/* * Copyright © 2018 Canonical Ltd. * * 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 . */ #include #include "cc-input-row.h" #include "cc-input-source-ibus.h" struct _CcInputRow { AdwActionRow parent_instance; CcInputSource *source; GtkListBox *drag_widget; GtkDragSource *drag_source; gdouble drag_x; gdouble drag_y; }; G_DEFINE_TYPE (CcInputRow, cc_input_row, ADW_TYPE_ACTION_ROW) enum { SIGNAL_SHOW_SETTINGS, SIGNAL_SHOW_LAYOUT, SIGNAL_MOVE_ROW, SIGNAL_REMOVE_ROW, SIGNAL_LAST }; static guint signals[SIGNAL_LAST] = { 0, }; static GdkContentProvider * drag_prepare_cb (GtkDragSource *source, double x, double y, CcInputRow *self) { self->drag_x = x; self->drag_y = y; return gdk_content_provider_new_typed (CC_TYPE_INPUT_ROW, self); } static void drag_begin_cb (GtkDragSource *source, GdkDrag *drag, CcInputRow *self) { GtkAllocation alloc; CcInputRow *drag_row; GtkWidget *drag_icon; gtk_widget_get_allocation (GTK_WIDGET (self), &alloc); self->drag_widget = GTK_LIST_BOX (gtk_list_box_new ()); gtk_widget_set_size_request (GTK_WIDGET (self->drag_widget), alloc.width, alloc.height); drag_row = cc_input_row_new (self->source); gtk_list_box_append (self->drag_widget, GTK_WIDGET (drag_row)); gtk_list_box_drag_highlight_row (self->drag_widget, GTK_LIST_BOX_ROW (drag_row)); drag_icon = gtk_drag_icon_get_for_drag (drag); gtk_drag_icon_set_child (GTK_DRAG_ICON (drag_icon), GTK_WIDGET (self->drag_widget)); gdk_drag_set_hotspot (drag, self->drag_x, self->drag_y); } static gboolean drop_cb (GtkDropTarget *drop_target, const GValue *value, gdouble x, gdouble y, CcInputRow *self) { CcInputRow *source; if (!G_VALUE_HOLDS (value, CC_TYPE_INPUT_ROW)) return FALSE; source = g_value_get_object (value); g_signal_emit (source, signals[SIGNAL_MOVE_ROW], 0, self); return TRUE; } static void move_up_cb (GtkWidget *widget, const char *action_name, GVariant *parameter) { CcInputRow *self = CC_INPUT_ROW (widget); GtkListBox *list_box = GTK_LIST_BOX (gtk_widget_get_parent (GTK_WIDGET (self))); gint previous_idx = gtk_list_box_row_get_index (GTK_LIST_BOX_ROW (self)) - 1; GtkListBoxRow *previous_row = gtk_list_box_get_row_at_index (list_box, previous_idx); if (previous_row == NULL) return; g_signal_emit (self, signals[SIGNAL_MOVE_ROW], 0, previous_row); } static void move_down_cb (GtkWidget *widget, const char *action_name, GVariant *parameter) { CcInputRow *self = CC_INPUT_ROW (widget); GtkListBox *list_box = GTK_LIST_BOX (gtk_widget_get_parent (GTK_WIDGET (self))); gint next_idx = gtk_list_box_row_get_index (GTK_LIST_BOX_ROW (self)) + 1; GtkListBoxRow *next_row = gtk_list_box_get_row_at_index (list_box, next_idx); if (next_row == NULL) return; g_signal_emit (next_row, signals[SIGNAL_MOVE_ROW], 0, self); } static void show_settings_cb (GtkWidget *widget, const char *action_name, GVariant *parameter) { CcInputRow *self = CC_INPUT_ROW (widget); g_signal_emit (self, signals[SIGNAL_SHOW_SETTINGS], 0); } static void show_layout_cb (GtkWidget *widget, const char *action_name, GVariant *parameter) { CcInputRow *self = CC_INPUT_ROW (widget); g_signal_emit (self, signals[SIGNAL_SHOW_LAYOUT], 0); } static void remove_cb (GtkWidget *widget, const char *action_name, GVariant *parameter) { CcInputRow *self = CC_INPUT_ROW (widget); g_signal_emit (self, signals[SIGNAL_REMOVE_ROW], 0); } static void cc_input_row_dispose (GObject *object) { CcInputRow *self = CC_INPUT_ROW (object); g_clear_object (&self->source); G_OBJECT_CLASS (cc_input_row_parent_class)->dispose (object); } void cc_input_row_class_init (CcInputRowClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); object_class->dispose = cc_input_row_dispose; gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/control-center/keyboard/cc-input-row.ui"); signals[SIGNAL_SHOW_SETTINGS] = g_signal_new ("show-settings", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0); signals[SIGNAL_SHOW_LAYOUT] = g_signal_new ("show-layout", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0); signals[SIGNAL_MOVE_ROW] = g_signal_new ("move-row", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, CC_TYPE_INPUT_ROW); signals[SIGNAL_REMOVE_ROW] = g_signal_new ("remove-row", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0); gtk_widget_class_install_action (widget_class, "row.move-up", NULL, move_up_cb); gtk_widget_class_install_action (widget_class, "row.move-down", NULL, move_down_cb); gtk_widget_class_install_action (widget_class, "row.show-layout", NULL, show_layout_cb); gtk_widget_class_install_action (widget_class, "row.show-settings", NULL, show_settings_cb); gtk_widget_class_install_action (widget_class, "row.remove", NULL, remove_cb); } void cc_input_row_init (CcInputRow *self) { GtkDropTarget *drop_target; gtk_widget_init_template (GTK_WIDGET (self)); self->drag_source = gtk_drag_source_new (); gtk_drag_source_set_actions (self->drag_source, GDK_ACTION_MOVE); g_signal_connect (self->drag_source, "prepare", G_CALLBACK (drag_prepare_cb), self); g_signal_connect (self->drag_source, "drag-begin", G_CALLBACK (drag_begin_cb), self); gtk_widget_add_controller (GTK_WIDGET (self), GTK_EVENT_CONTROLLER (self->drag_source)); drop_target = gtk_drop_target_new (CC_TYPE_INPUT_ROW, GDK_ACTION_MOVE); gtk_drop_target_set_preload (drop_target, TRUE); g_signal_connect (drop_target, "drop", G_CALLBACK (drop_cb), self); gtk_widget_add_controller (GTK_WIDGET (self), GTK_EVENT_CONTROLLER (drop_target)); } static void label_changed_cb (CcInputRow *self) { g_autofree gchar *label = cc_input_source_get_label (self->source); adw_preferences_row_set_title (ADW_PREFERENCES_ROW (self), label); } CcInputRow * cc_input_row_new (CcInputSource *source) { CcInputRow *self; self = g_object_new (CC_TYPE_INPUT_ROW, NULL); self->source = g_object_ref (source); g_signal_connect_object (source, "label-changed", G_CALLBACK (label_changed_cb), self, G_CONNECT_SWAPPED); label_changed_cb (self); gtk_widget_action_set_enabled (GTK_WIDGET (self), "row.show-settings", CC_IS_INPUT_SOURCE_IBUS (source)); return self; } CcInputSource * cc_input_row_get_source (CcInputRow *self) { g_return_val_if_fail (CC_IS_INPUT_ROW (self), NULL); return self->source; } void cc_input_row_set_removable (CcInputRow *self, gboolean removable) { g_return_if_fail (CC_IS_INPUT_ROW (self)); gtk_widget_action_set_enabled (GTK_WIDGET (self), "row.remove", removable); } void cc_input_row_set_draggable (CcInputRow *self, gboolean draggable) { gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (self->drag_source), draggable ? GTK_PHASE_BUBBLE : GTK_PHASE_NONE); }