summaryrefslogtreecommitdiffstats
path: root/panels/wacom/cc-wacom-stylus-page.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--panels/wacom/cc-wacom-stylus-page.c503
1 files changed, 503 insertions, 0 deletions
diff --git a/panels/wacom/cc-wacom-stylus-page.c b/panels/wacom/cc-wacom-stylus-page.c
new file mode 100644
index 0000000..2fc6c5a
--- /dev/null
+++ b/panels/wacom/cc-wacom-stylus-page.c
@@ -0,0 +1,503 @@
+/*
+ * Copyright © 2011 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, see <http://www.gnu.org/licenses/>.
+ *
+ * Authors: Peter Hutterer <peter.hutterer@redhat.com>
+ * Bastien Nocera <hadess@hadess.net>
+ *
+ */
+
+#include <config.h>
+
+#include <glib/gi18n.h>
+#include "cc-wacom-stylus-page.h"
+#include "cc-wacom-nav-button.h"
+#include <gtk/gtk.h>
+#include <gdesktop-enums.h>
+
+#include <string.h>
+
+#define WID(x) (GtkWidget *) gtk_builder_get_object (page->builder, x)
+#define CWID(x) (GtkContainer *) gtk_builder_get_object (page->builder, x)
+
+struct _CcWacomStylusPage
+{
+ GtkBox parent_instance;
+
+ CcWacomTool *stylus;
+ GtkBuilder *builder;
+ GtkWidget *nav;
+ GSettings *stylus_settings;
+};
+
+G_DEFINE_TYPE (CcWacomStylusPage, cc_wacom_stylus_page, GTK_TYPE_BOX)
+
+/* Button combo box storage columns */
+enum {
+ BUTTONNUMBER_COLUMN,
+ BUTTONNAME_COLUMN,
+ N_BUTTONCOLUMNS
+};
+
+/* GSettings stores pressurecurve as 4 values like the driver. We map slider
+ * scale to these values given the array below. These settings were taken from
+ * wacomcpl, where they've been around for years.
+ */
+#define N_PRESSURE_CURVES 7
+static const gint32 PRESSURE_CURVES[N_PRESSURE_CURVES][4] = {
+ { 0, 75, 25, 100 }, /* soft */
+ { 0, 50, 50, 100 },
+ { 0, 25, 75, 100 },
+ { 0, 0, 100, 100 }, /* neutral */
+ { 25, 0, 100, 75 },
+ { 50, 0, 100, 50 },
+ { 75, 0, 100, 25 } /* firm */
+};
+
+static void
+set_pressurecurve (GtkRange *range, GSettings *settings, const gchar *key)
+{
+ gint slider_val = gtk_range_get_value (range);
+ GVariant *values[4],
+ *array;
+ int i;
+
+ for (i = 0; i < G_N_ELEMENTS (values); i++)
+ values[i] = g_variant_new_int32 (PRESSURE_CURVES[slider_val][i]);
+
+ array = g_variant_new_array (G_VARIANT_TYPE_INT32, values, G_N_ELEMENTS (values));
+
+ g_settings_set_value (settings, key, array);
+}
+
+static void
+tip_feel_value_changed_cb (CcWacomStylusPage *page)
+{
+ set_pressurecurve (GTK_RANGE (WID ("scale-tip-feel")), page->stylus_settings, "pressure-curve");
+}
+
+static void
+eraser_feel_value_changed_cb (CcWacomStylusPage *page)
+{
+ set_pressurecurve (GTK_RANGE (WID ("scale-eraser-feel")), page->stylus_settings, "eraser-pressure-curve");
+}
+
+static void
+set_feel_from_gsettings (GtkAdjustment *adjustment, GSettings *settings, const gchar *key)
+{
+ GVariant *variant;
+ const gint32 *values;
+ gsize nvalues;
+ int i;
+
+ variant = g_settings_get_value (settings, key);
+ values = g_variant_get_fixed_array (variant, &nvalues, sizeof (gint32));
+
+ if (nvalues != 4) {
+ g_warning ("Invalid pressure curve format, expected 4 values (got %"G_GSIZE_FORMAT")", nvalues);
+ return;
+ }
+
+ for (i = 0; i < N_PRESSURE_CURVES; i++) {
+ if (memcmp (PRESSURE_CURVES[i], values, sizeof (gint32) * 4) == 0) {
+ gtk_adjustment_set_value (adjustment, i);
+ break;
+ }
+ }
+}
+
+static void
+set_button_mapping_from_gsettings (GtkComboBox *combo, GSettings* settings, const gchar *key)
+{
+ GDesktopStylusButtonAction action;
+ GtkTreeModel *model;
+ GtkTreeIter iter;
+ gboolean valid;
+
+ action = g_settings_get_enum (settings, key);
+ model = gtk_combo_box_get_model (combo);
+ valid = gtk_tree_model_get_iter_first (model, &iter);
+
+ while (valid) {
+ gint button;
+
+ gtk_tree_model_get (model, &iter,
+ BUTTONNUMBER_COLUMN, &button,
+ -1);
+
+ /* Currently button values match logical X buttons. If we
+ * introduce things like double-click, this code must
+ * change. Recommendation: use negative buttons numbers for
+ * special ones.
+ */
+
+ if (button == action) {
+ gtk_combo_box_set_active_iter (combo, &iter);
+ break;
+ }
+
+ valid = gtk_tree_model_iter_next (model, &iter);
+ }
+}
+
+static void
+button_changed_cb (CcWacomStylusPage *page)
+{
+ GtkTreeIter iter;
+ GtkListStore *liststore;
+ gint mapping_b2,
+ mapping_b3,
+ mapping_b4;
+
+ if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (WID ("combo-bottombutton")), &iter))
+ return;
+
+ liststore = GTK_LIST_STORE (WID ("liststore-buttons"));
+ gtk_tree_model_get (GTK_TREE_MODEL (liststore), &iter,
+ BUTTONNUMBER_COLUMN, &mapping_b2,
+ -1);
+
+ if (cc_wacom_tool_get_num_buttons (page->stylus) > 1) {
+ if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (WID ("combo-topbutton")), &iter))
+ return;
+
+ gtk_tree_model_get (GTK_TREE_MODEL (liststore), &iter,
+ BUTTONNUMBER_COLUMN, &mapping_b3,
+ -1);
+ } else {
+ mapping_b3 = 0;
+ }
+
+ if (cc_wacom_tool_get_num_buttons (page->stylus) > 2) {
+ if (!gtk_combo_box_get_active_iter (GTK_COMBO_BOX (WID ("combo-thirdbutton")), &iter))
+ return;
+
+ gtk_tree_model_get (GTK_TREE_MODEL (liststore), &iter,
+ BUTTONNUMBER_COLUMN, &mapping_b4,
+ -1);
+ } else {
+ mapping_b4 = 0;
+ }
+
+ g_settings_set_enum (page->stylus_settings, "button-action", mapping_b2);
+ g_settings_set_enum (page->stylus_settings, "secondary-button-action", mapping_b3);
+ g_settings_set_enum (page->stylus_settings, "tertiary-button-action", mapping_b4);
+}
+
+static void
+combobox_text_cellrenderer (GtkComboBox *combo, int name_column)
+{
+ GtkCellRenderer *renderer;
+
+ renderer = gtk_cell_renderer_text_new ();
+ gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, TRUE);
+ gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo), renderer,
+ "text", BUTTONNAME_COLUMN, NULL);
+}
+
+/* Boilerplate code goes below */
+
+static void
+cc_wacom_stylus_page_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ switch (property_id)
+ {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ }
+}
+
+static void
+cc_wacom_stylus_page_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ switch (property_id)
+ {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ }
+}
+
+static void
+cc_wacom_stylus_page_dispose (GObject *object)
+{
+ CcWacomStylusPage *page = CC_WACOM_STYLUS_PAGE (object);
+
+ g_clear_object (&page->builder);
+
+ G_OBJECT_CLASS (cc_wacom_stylus_page_parent_class)->dispose (object);
+}
+
+static void
+cc_wacom_stylus_page_class_init (CcWacomStylusPageClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->get_property = cc_wacom_stylus_page_get_property;
+ object_class->set_property = cc_wacom_stylus_page_set_property;
+ object_class->dispose = cc_wacom_stylus_page_dispose;
+}
+
+static void
+add_marks (GtkScale *scale)
+{
+ gint i;
+
+ for (i = 0; i < N_PRESSURE_CURVES; i++)
+ gtk_scale_add_mark (scale, i, GTK_POS_BOTTOM, NULL);
+}
+
+static void
+cc_wacom_stylus_page_init (CcWacomStylusPage *page)
+{
+ g_autoptr(GError) error = NULL;
+ GtkComboBox *combo;
+ GtkWidget *box;
+ char *objects[] = {
+ "stylus-grid",
+ "liststore-buttons",
+ "adjustment-tip-feel",
+ "adjustment-eraser-feel",
+ NULL
+ };
+
+ page->builder = gtk_builder_new ();
+
+ gtk_builder_add_objects_from_resource (page->builder,
+ "/org/gnome/control-center/wacom/wacom-stylus-page.ui",
+ objects,
+ &error);
+ if (error != NULL) {
+ g_warning ("Error loading UI file: %s", error->message);
+ return;
+ }
+
+ box = WID ("stylus-grid");
+ gtk_container_add (GTK_CONTAINER (page), box);
+ gtk_widget_set_vexpand (GTK_WIDGET (box), TRUE);
+
+ add_marks (GTK_SCALE (WID ("scale-tip-feel")));
+ add_marks (GTK_SCALE (WID ("scale-eraser-feel")));
+
+ g_signal_connect_object (WID ("scale-tip-feel"), "value-changed",
+ G_CALLBACK (tip_feel_value_changed_cb), page, G_CONNECT_SWAPPED);
+ g_signal_connect_object (WID ("scale-eraser-feel"), "value-changed",
+ G_CALLBACK (eraser_feel_value_changed_cb), page, G_CONNECT_SWAPPED);
+
+ combo = GTK_COMBO_BOX (WID ("combo-topbutton"));
+ combobox_text_cellrenderer (combo, BUTTONNAME_COLUMN);
+ g_signal_connect_object (combo, "changed",
+ G_CALLBACK (button_changed_cb), page, G_CONNECT_SWAPPED);
+
+ combo = GTK_COMBO_BOX (WID ("combo-bottombutton"));
+ combobox_text_cellrenderer (combo, BUTTONNAME_COLUMN);
+ g_signal_connect_object (combo, "changed",
+ G_CALLBACK (button_changed_cb), page, G_CONNECT_SWAPPED);
+
+ combo = GTK_COMBO_BOX (WID ("combo-thirdbutton"));
+ combobox_text_cellrenderer (combo, BUTTONNAME_COLUMN);
+ g_signal_connect_object (G_OBJECT (combo), "changed",
+ G_CALLBACK (button_changed_cb), page, G_CONNECT_SWAPPED);
+
+ page->nav = cc_wacom_nav_button_new ();
+ gtk_widget_set_halign (page->nav, GTK_ALIGN_END);
+ gtk_widget_set_margin_start (page->nav, 10);
+ gtk_widget_show (page->nav);
+ gtk_container_add (CWID ("navigation-placeholder"), page->nav);
+}
+
+static void
+set_icon_name (CcWacomStylusPage *page,
+ const char *widget_name,
+ const char *icon_name)
+{
+ g_autofree gchar *resource = NULL;
+
+ resource = g_strdup_printf ("/org/gnome/control-center/wacom/%s.svg", icon_name);
+ gtk_image_set_from_resource (GTK_IMAGE (WID (widget_name)), resource);
+}
+
+/* Different types of layout for the stylus config */
+enum {
+ LAYOUT_NORMAL, /* eraser, 2 buttons, tip */
+ LAYOUT_INKING, /* tip */
+ LAYOUT_AIRBRUSH, /* eraser, 1 button, tip */
+ LAYOUT_GENERIC_2_BUTTONS_NO_ERASER, /* 2 buttons, tip, no eraser */
+ LAYOUT_3DPEN, /* 3 buttons, tip, no eraser */
+ LAYOUT_OTHER
+};
+
+static void
+remove_buttons (CcWacomStylusPage *page, int n)
+{
+ if (n < 3) {
+ gtk_widget_destroy (WID ("combo-thirdbutton"));
+ gtk_widget_destroy (WID ("label-third-button"));
+ }
+ if (n < 2) {
+ gtk_widget_destroy (WID ("combo-topbutton"));
+ gtk_widget_destroy (WID ("label-top-button"));
+ gtk_label_set_text (GTK_LABEL (WID ("label-lower-button")), _("Button"));
+ }
+ if (n < 1) {
+ gtk_widget_destroy (WID ("combo-bottombutton"));
+ gtk_widget_destroy (WID ("label-lower-button"));
+ }
+}
+
+static void
+remove_eraser (CcWacomStylusPage *page)
+{
+ gtk_widget_destroy (WID ("eraser-box"));
+ gtk_widget_destroy (WID ("label-eraser-feel"));
+}
+
+static void
+update_stylus_ui (CcWacomStylusPage *page,
+ int layout)
+{
+ switch (layout) {
+ case LAYOUT_NORMAL:
+ remove_buttons (page, 2);
+ break;
+ case LAYOUT_INKING:
+ remove_buttons (page, 0);
+ remove_eraser (page);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("label-tip-feel"),
+ "top_attach", 0, NULL);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("box-tip-feel"),
+ "top_attach", 0, NULL);
+ break;
+ case LAYOUT_AIRBRUSH:
+ remove_buttons (page, 1);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("label-lower-button"),
+ "top_attach", 1, NULL);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("combo-bottombutton"),
+ "top_attach", 1, NULL);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("label-tip-feel"),
+ "top_attach", 2, NULL);
+ gtk_container_child_set (CWID ("stylus-controls-grid"),
+ WID ("box-tip-feel"),
+ "top_attach", 2, NULL);
+ break;
+ case LAYOUT_GENERIC_2_BUTTONS_NO_ERASER:
+ remove_buttons (page, 2);
+ remove_eraser (page);
+ break;
+ case LAYOUT_3DPEN:
+ remove_buttons (page, 3);
+ remove_eraser (page);
+ break;
+ case LAYOUT_OTHER:
+ /* We already warn about it in cc_wacom_stylus_page_new () */
+ break;
+ }
+}
+
+GtkWidget *
+cc_wacom_stylus_page_new (CcWacomTool *stylus)
+{
+ CcWacomStylusPage *page;
+ guint num_buttons;
+ int layout;
+ gboolean has_eraser;
+
+ g_return_val_if_fail (CC_IS_WACOM_TOOL (stylus), NULL);
+
+ page = g_object_new (CC_TYPE_WACOM_STYLUS_PAGE, NULL);
+
+ page->stylus = stylus;
+
+ /* Icon */
+ set_icon_name (page, "image-stylus", cc_wacom_tool_get_icon_name (stylus));
+
+ /* Settings */
+ page->stylus_settings = cc_wacom_tool_get_settings (stylus);
+ has_eraser = cc_wacom_tool_get_has_eraser (stylus);
+
+ /* Stylus name */
+ gtk_label_set_text (GTK_LABEL (WID ("label-stylus")), cc_wacom_tool_get_name (stylus));
+
+ num_buttons = cc_wacom_tool_get_num_buttons (stylus);
+ if (num_buttons == 0 && !has_eraser)
+ layout = LAYOUT_INKING;
+ else if (num_buttons == 2 && has_eraser)
+ layout = LAYOUT_NORMAL;
+ else if (num_buttons == 1 && has_eraser)
+ layout = LAYOUT_AIRBRUSH;
+ else if (num_buttons == 2 && !has_eraser)
+ layout = LAYOUT_GENERIC_2_BUTTONS_NO_ERASER;
+ else if (num_buttons == 3 && !has_eraser)
+ layout = LAYOUT_3DPEN;
+ else {
+ layout = LAYOUT_OTHER;
+ remove_buttons (page, num_buttons);
+
+ /* Gray out eraser if not available */
+ gtk_widget_set_sensitive (WID ("eraser-box"), has_eraser);
+ gtk_widget_set_sensitive (WID ("label-eraser-feel"), has_eraser);
+
+ g_warning ("The layout of this page is not known, %d buttons, %s eraser",
+ num_buttons, has_eraser ? "with" : "without");
+ }
+
+ update_stylus_ui (page, layout);
+
+ if (num_buttons >= 3)
+ set_button_mapping_from_gsettings (GTK_COMBO_BOX (WID ("combo-thirdbutton")),
+ page->stylus_settings, "tertiary-button-action");
+ if (num_buttons >= 2)
+ set_button_mapping_from_gsettings (GTK_COMBO_BOX (WID ("combo-topbutton")),
+ page->stylus_settings, "secondary-button-action");
+ if (num_buttons >= 1)
+ set_button_mapping_from_gsettings (GTK_COMBO_BOX (WID ("combo-bottombutton")),
+ page->stylus_settings, "button-action");
+ set_feel_from_gsettings (GTK_ADJUSTMENT (WID ("adjustment-tip-feel")),
+ page->stylus_settings, "pressure-curve");
+
+ if (has_eraser)
+ set_feel_from_gsettings (GTK_ADJUSTMENT (WID ("adjustment-eraser-feel")),
+ page->stylus_settings, "eraser-pressure-curve");
+
+ return GTK_WIDGET (page);
+}
+
+CcWacomTool *
+cc_wacom_stylus_page_get_tool (CcWacomStylusPage *page)
+{
+ return page->stylus;
+}
+
+void
+cc_wacom_stylus_page_set_navigation (CcWacomStylusPage *page,
+ GtkNotebook *notebook)
+{
+ g_return_if_fail (CC_IS_WACOM_STYLUS_PAGE (page));
+
+ g_object_set (G_OBJECT (page->nav),
+ "notebook", notebook,
+ "ignore-first", TRUE,
+ NULL);
+}