/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- * * Copyright (C) 2001, 2012 Red Hat, Inc. * Copyright (C) 2001 Ximian, Inc. * * Written by: Jonathon Blandford , * Bradford Hovinen , * Ondrej Holy , * * 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, 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, see . */ #include #include #include #include #include #include #include #include #include "gnome-mouse-properties.h" #include "gsd-input-helper.h" #include "gsd-device-manager.h" #include "list-box-helper.h" #include "cc-mouse-caps-helper.h" #include #include #include struct _CcMouseProperties { GtkBin parent_instance; GtkWidget *edge_scrolling_row; GtkWidget *edge_scrolling_switch; GtkWidget *general_listbox; GtkWidget *mouse_frame; GtkWidget *mouse_listbox; GtkWidget *mouse_natural_scrolling_switch; GtkWidget *mouse_speed_scale; GtkWidget *primary_button_left; GtkWidget *primary_button_right; GtkWidget *scrolled_window; GtkWidget *tap_to_click_row; GtkWidget *tap_to_click_switch; GtkWidget *touchpad_frame; GtkWidget *touchpad_listbox; GtkWidget *touchpad_natural_scrolling_switch; GtkWidget *touchpad_options_listbox; GtkWidget *touchpad_speed_scale; GtkWidget *touchpad_toggle_switch; GtkWidget *two_finger_scrolling_row; GtkWidget *two_finger_scrolling_switch; GSettings *mouse_settings; GSettings *gsd_mouse_settings; GSettings *touchpad_settings; GsdDeviceManager *device_manager; guint device_added_id; guint device_removed_id; gboolean have_mouse; gboolean have_touchpad; gboolean have_touchscreen; gboolean have_synaptics; gboolean left_handed; GtkGesture *left_gesture; GtkGesture *right_gesture; gboolean changing_scroll; }; G_DEFINE_TYPE (CcMouseProperties, cc_mouse_properties, GTK_TYPE_BIN); static void setup_touchpad_options (CcMouseProperties *self) { gboolean edge_scroll_enabled; gboolean two_finger_scroll_enabled; gboolean have_two_finger_scrolling; gboolean have_edge_scrolling; gboolean have_tap_to_click; if (self->have_synaptics || !self->have_touchpad) { gtk_widget_hide (self->touchpad_frame); return; } cc_touchpad_check_capabilities (&have_two_finger_scrolling, &have_edge_scrolling, &have_tap_to_click); gtk_widget_show (self->touchpad_frame); gtk_widget_set_visible (self->two_finger_scrolling_row, have_two_finger_scrolling); gtk_widget_set_visible (self->edge_scrolling_row, have_edge_scrolling); gtk_widget_set_visible (self->tap_to_click_row, have_tap_to_click); edge_scroll_enabled = g_settings_get_boolean (self->touchpad_settings, "edge-scrolling-enabled"); two_finger_scroll_enabled = g_settings_get_boolean (self->touchpad_settings, "two-finger-scrolling-enabled"); if (edge_scroll_enabled && two_finger_scroll_enabled) { /* You cunning user set both, but you can only have one set in that UI */ self->changing_scroll = TRUE; gtk_switch_set_active (GTK_SWITCH (self->two_finger_scrolling_switch), two_finger_scroll_enabled); self->changing_scroll = FALSE; gtk_switch_set_active (GTK_SWITCH (self->edge_scrolling_switch), FALSE); } else { self->changing_scroll = TRUE; gtk_switch_set_active (GTK_SWITCH (self->edge_scrolling_switch), edge_scroll_enabled); gtk_switch_set_active (GTK_SWITCH (self->two_finger_scrolling_switch), two_finger_scroll_enabled); self->changing_scroll = FALSE; } } static void two_finger_scrolling_changed_event (CcMouseProperties *self, gboolean state) { if (self->changing_scroll) return; g_settings_set_boolean (self->touchpad_settings, "two-finger-scrolling-enabled", state); gtk_switch_set_state (GTK_SWITCH (self->two_finger_scrolling_switch), state); if (state && gtk_widget_get_visible (self->edge_scrolling_row)) { /* Disable edge scrolling if two-finger scrolling is enabled */ gtk_switch_set_state (GTK_SWITCH (self->edge_scrolling_switch), FALSE); } } static void edge_scrolling_changed_event (CcMouseProperties *self, gboolean state) { if (self->changing_scroll) return; g_settings_set_boolean (self->touchpad_settings, "edge-scrolling-enabled", state); gtk_switch_set_state (GTK_SWITCH (self->edge_scrolling_switch), state); if (state && gtk_widget_get_visible (self->two_finger_scrolling_row)) { /* Disable two-finger scrolling if edge scrolling is enabled */ gtk_switch_set_state (GTK_SWITCH (self->two_finger_scrolling_switch), FALSE); } } static gboolean get_touchpad_enabled (GSettings *settings) { GDesktopDeviceSendEvents send_events; send_events = g_settings_get_enum (settings, "send-events"); return send_events == G_DESKTOP_DEVICE_SEND_EVENTS_ENABLED; } static gboolean show_touchpad_enabling_switch (CcMouseProperties *self) { if (!self->have_touchpad) return FALSE; g_debug ("Should we show the touchpad disable switch: have_mouse: %s have_touchscreen: %s\n", self->have_mouse ? "true" : "false", self->have_touchscreen ? "true" : "false"); /* Let's show the button when a mouse or touchscreen is present */ if (self->have_mouse || self->have_touchscreen) return TRUE; /* Let's also show when the touchpad is disabled. */ if (!get_touchpad_enabled (self->touchpad_settings)) return TRUE; return FALSE; } static gboolean touchpad_enabled_get_mapping (GValue *value, GVariant *variant, gpointer user_data) { gboolean enabled; enabled = g_strcmp0 (g_variant_get_string (variant, NULL), "enabled") == 0; g_value_set_boolean (value, enabled); return TRUE; } static GVariant * touchpad_enabled_set_mapping (const GValue *value, const GVariantType *type, gpointer user_data) { gboolean enabled; enabled = g_value_get_boolean (value); return g_variant_new_string (enabled ? "enabled" : "disabled"); } static void handle_secondary_button (CcMouseProperties *self, GtkWidget *button, GtkGesture *gesture) { gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (gesture), FALSE); gtk_gesture_single_set_exclusive (GTK_GESTURE_SINGLE (gesture), TRUE); gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (gesture), GDK_BUTTON_SECONDARY); g_signal_connect_swapped (gesture, "pressed", G_CALLBACK (gtk_button_clicked), button); gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (gesture), GTK_PHASE_BUBBLE); } /* Set up the property editors in the dialog. */ static void setup_dialog (CcMouseProperties *self) { GtkWidget *button; self->left_handed = g_settings_get_boolean (self->mouse_settings, "left-handed"); button = self->left_handed ? self->primary_button_right : self->primary_button_left; gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); g_settings_bind (self->mouse_settings, "left-handed", self->primary_button_left, "active", G_SETTINGS_BIND_DEFAULT | G_SETTINGS_BIND_INVERT_BOOLEAN); g_settings_bind (self->mouse_settings, "left-handed", self->primary_button_right, "active", G_SETTINGS_BIND_DEFAULT); /* Allow changing orientation with either button */ button = self->primary_button_right; self->right_gesture = gtk_gesture_multi_press_new (button); handle_secondary_button (self, button, self->right_gesture); button = self->primary_button_left; self->left_gesture = gtk_gesture_multi_press_new (button); handle_secondary_button (self, button, self->left_gesture); g_settings_bind (self->mouse_settings, "natural-scroll", self->mouse_natural_scrolling_switch, "active", G_SETTINGS_BIND_DEFAULT); gtk_list_box_set_header_func (GTK_LIST_BOX (self->general_listbox), cc_list_box_update_header_func, NULL, NULL); gtk_list_box_set_header_func (GTK_LIST_BOX (self->touchpad_listbox), cc_list_box_update_header_func, NULL, NULL); /* Mouse section */ gtk_widget_set_visible (self->mouse_frame, self->have_mouse); g_settings_bind (self->mouse_settings, "speed", gtk_range_get_adjustment (GTK_RANGE (self->mouse_speed_scale)), "value", G_SETTINGS_BIND_DEFAULT); gtk_list_box_set_header_func (GTK_LIST_BOX (self->mouse_listbox), cc_list_box_update_header_func, NULL, NULL); /* Touchpad section */ gtk_widget_set_visible (self->touchpad_toggle_switch, show_touchpad_enabling_switch (self)); g_settings_bind_with_mapping (self->touchpad_settings, "send-events", self->touchpad_toggle_switch, "active", G_SETTINGS_BIND_DEFAULT, touchpad_enabled_get_mapping, touchpad_enabled_set_mapping, NULL, NULL); g_settings_bind_with_mapping (self->touchpad_settings, "send-events", self->touchpad_options_listbox, "sensitive", G_SETTINGS_BIND_GET, touchpad_enabled_get_mapping, touchpad_enabled_set_mapping, NULL, NULL); g_settings_bind (self->touchpad_settings, "natural-scroll", self->touchpad_natural_scrolling_switch, "active", G_SETTINGS_BIND_DEFAULT); g_settings_bind (self->touchpad_settings, "speed", gtk_range_get_adjustment (GTK_RANGE (self->touchpad_speed_scale)), "value", G_SETTINGS_BIND_DEFAULT); g_settings_bind (self->touchpad_settings, "tap-to-click", self->tap_to_click_switch, "active", G_SETTINGS_BIND_DEFAULT); setup_touchpad_options (self); gtk_list_box_set_header_func (GTK_LIST_BOX (self->touchpad_options_listbox), cc_list_box_update_header_func, NULL, NULL); } /* Callback issued when a button is clicked on the dialog */ static void device_changed (GsdDeviceManager *device_manager, GsdDevice *device, CcMouseProperties *self) { self->have_touchpad = touchpad_is_present (); setup_touchpad_options (self); self->have_mouse = mouse_is_present (); gtk_widget_set_visible (self->mouse_frame, self->have_mouse); gtk_widget_set_visible (self->touchpad_toggle_switch, show_touchpad_enabling_switch (self)); } static void on_content_size_changed (CcMouseProperties *self, GtkAllocation *allocation) { if (allocation->height < 490) { gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (self->scrolled_window), GTK_POLICY_NEVER, GTK_POLICY_NEVER); } else { gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (self->scrolled_window), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_min_content_height (GTK_SCROLLED_WINDOW (self->scrolled_window), 490); } } static void cc_mouse_properties_finalize (GObject *object) { CcMouseProperties *self = CC_MOUSE_PROPERTIES (object); g_clear_object (&self->mouse_settings); g_clear_object (&self->gsd_mouse_settings); g_clear_object (&self->touchpad_settings); g_clear_object (&self->right_gesture); g_clear_object (&self->left_gesture); if (self->device_manager != NULL) { g_signal_handler_disconnect (self->device_manager, self->device_added_id); self->device_added_id = 0; g_signal_handler_disconnect (self->device_manager, self->device_removed_id); self->device_removed_id = 0; self->device_manager = NULL; } G_OBJECT_CLASS (cc_mouse_properties_parent_class)->finalize (object); } static void cc_mouse_properties_class_init (CcMousePropertiesClass *klass) { GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = cc_mouse_properties_finalize; gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/control-center/mouse/gnome-mouse-properties.ui"); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, edge_scrolling_row); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, edge_scrolling_switch); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, general_listbox); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, mouse_frame); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, mouse_listbox); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, mouse_natural_scrolling_switch); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, mouse_speed_scale); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, primary_button_left); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, primary_button_right); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, scrolled_window); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, tap_to_click_row); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, tap_to_click_switch); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_frame); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_listbox); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_natural_scrolling_switch); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_options_listbox); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_speed_scale); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, touchpad_toggle_switch); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, two_finger_scrolling_row); gtk_widget_class_bind_template_child (widget_class, CcMouseProperties, two_finger_scrolling_switch); gtk_widget_class_bind_template_callback (widget_class, edge_scrolling_changed_event); gtk_widget_class_bind_template_callback (widget_class, two_finger_scrolling_changed_event); gtk_widget_class_bind_template_callback (widget_class, on_content_size_changed); } static void cc_mouse_properties_init (CcMouseProperties *self) { g_autoptr(GError) error = NULL; gtk_widget_init_template (GTK_WIDGET (self)); self->mouse_settings = g_settings_new ("org.gnome.desktop.peripherals.mouse"); self->gsd_mouse_settings = g_settings_new ("org.gnome.settings-daemon.peripherals.mouse"); self->touchpad_settings = g_settings_new ("org.gnome.desktop.peripherals.touchpad"); self->device_manager = gsd_device_manager_get (); self->device_added_id = g_signal_connect (self->device_manager, "device-added", G_CALLBACK (device_changed), self); self->device_removed_id = g_signal_connect (self->device_manager, "device-removed", G_CALLBACK (device_changed), self); self->have_mouse = mouse_is_present (); self->have_touchpad = touchpad_is_present (); self->have_touchscreen = touchscreen_is_present (); self->have_synaptics = cc_synaptics_check (); if (self->have_synaptics) g_warning ("Detected synaptics X driver, please migrate to libinput"); self->changing_scroll = FALSE; setup_dialog (self); } GtkWidget * cc_mouse_properties_new (void) { return (GtkWidget *) g_object_new (CC_TYPE_MOUSE_PROPERTIES, NULL); }