/* Chrysalide - Outil d'analyse de fichiers binaires * regedit.c - panneau d'affichage des paramètres de configuration * * Copyright (C) 2014-2019 Cyrille Bagard * * This file is part of Chrysalide. * * Chrysalide 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 3 of the License, or * (at your option) any later version. * * Chrysalide 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "regedit.h" #include <assert.h> #include <malloc.h> #include <regex.h> #include <stdarg.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <gtk/gtk.h> #include "../agroup.h" #include "../panel-int.h" #include "../../core/params.h" #include "../../common/cpp.h" #include "../../common/extstr.h" #include "../../gtkext/easygtk.h" #include "../../gtkext/named.h" /* -------------------------- PARTIE PRINCIPALE DU PANNEAU -------------------------- */ /* Panneau d'affichage des paramètres de configuration (instance) */ struct _GRegeditPanel { GPanelItem parent; /* A laisser en premier */ GGenConfig *config; /* Configuration à afficher */ regex_t *filter; /* Filtre appliqué ou NULL */ GtkMenu *menu; /* Menu contextuel pour param. */ }; /* Panneau d'affichage des paramètres de configuration (classe) */ struct _GRegeditPanelClass { GPanelItemClass parent; /* A laisser en premier */ }; /* Colonnes de la liste des messages */ typedef enum _CfgParamColumn { CPC_PARAM, /* Paramètre présenté */ CPC_BOLD, /* Visuel des changements */ CPC_PATH, /* Chemin d'accès à une valeur */ CPC_STATUS, /* Etat de la définition */ CPC_TYPE, /* Type de paramètre */ CPC_VALUE, /* Valeur courante */ LGC_COUNT /* Nombre de colonnes */ } CfgParamColumn; /* Initialise la classe des panneaux des paramètres de config. */ static void g_regedit_panel_class_init(GRegeditPanelClass *); /* Initialise une instance de panneau de paramètres de config. */ static void g_regedit_panel_init(GRegeditPanel *); /* Supprime toutes les références externes. */ static void g_regedit_panel_dispose(GRegeditPanel *); /* Procède à la libération totale de la mémoire. */ static void g_regedit_panel_finalize(GRegeditPanel *); /* Fournit le nom interne attribué à l'élément réactif. */ static char *g_regedit_panel_class_get_key(const GRegeditPanelClass *); /* Indique le chemin initial de la localisation d'un panneau. */ static char *g_regedit_panel_class_get_path(const GRegeditPanelClass *); /* ------------------------- AFFICHAGE A L'AIDE D'UNE LISTE ------------------------- */ /* Recharge une configuration donnée à l'affichage. */ static void reload_config_into_treeview(GRegeditPanel *); /* Actualise l'affichage des données d'un paramètre modifié. */ static void on_configuration_param_modified(GGenConfig *, GCfgParam *, GRegeditPanel *); /* Actualise la valeur affichée d'un paramètre de configuration. */ static void update_config_param_value(GtkListStore *, GtkTreeIter *); /* Etablit une comparaison entre deux lignes de paramètres. */ static gint compare_config_list_columns(GtkTreeModel *, GtkTreeIter *, GtkTreeIter *, gpointer); /* Réagit à une pression sur <Shift+F2> et simule l'édition. */ static gboolean on_key_pressed_over_params(GtkTreeView *, GdkEventKey *, GRegeditPanel *); /* Réagit à une édition de la valeur d'un paramètre. */ static void on_param_value_edited(GtkCellRendererText *, gchar *, gchar *, GRegeditPanel *); /* ------------------------- FILTRAGE DES SYMBOLES PRESENTS ------------------------- */ /* Démarre l'actualisation du filtrage des paramètres. */ static void on_param_search_changed(GtkSearchEntry *, GRegeditPanel *); /* Détermine si un paramètre doit être filtré ou non. */ static bool is_param_filtered(GRegeditPanel *, const char *); /* ------------------------ ATTRIBUTION D'UN MENU CONTEXTUEL ------------------------ */ /* Assure la gestion des clics de souris sur les paramètres. */ static gboolean on_button_press_over_params(GtkWidget *, GdkEventButton *, GRegeditPanel *); /* Construit le menu contextuel pour les paramètres. */ GtkMenu *build_param_panel_menu(GRegeditPanel *); /* Fournit le paramètre sélectionné dans la liste. */ static GCfgParam *get_selected_panel_param(GtkTreeView *, GtkTreeIter *); /* Réagit avec le menu "Copier le nom". */ static void mcb_param_panel_copy(GtkMenuItem *, GRegeditPanel *); /* Réagit avec le menu "Valeur néant". */ static void mcb_param_panel_empty(GtkMenuItem *, GRegeditPanel *); /* Réagit avec le menu "Réinitialiser la valeur". */ static void mcb_param_panel_reset(GtkMenuItem *, GRegeditPanel *); /* ---------------------------------------------------------------------------------- */ /* PARTIE PRINCIPALE DU PANNEAU */ /* ---------------------------------------------------------------------------------- */ /* Indique le type défini pour un panneau d'aperçu de graphiques. */ G_DEFINE_TYPE(GRegeditPanel, g_regedit_panel, G_TYPE_PANEL_ITEM); /****************************************************************************** * * * Paramètres : class = classe à initialiser. * * * * Description : Initialise la classe des panneaux des paramètres de config. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_regedit_panel_class_init(GRegeditPanelClass *class) { GObjectClass *object; /* Autre version de la classe */ GEditorItemClass *item; /* Encore une autre vision... */ GPanelItemClass *panel; /* Version parente de la classe*/ object = G_OBJECT_CLASS(class); object->dispose = (GObjectFinalizeFunc/* ! */)g_regedit_panel_dispose; object->finalize = (GObjectFinalizeFunc)g_regedit_panel_finalize; item = G_EDITOR_ITEM_CLASS(class); item->get_key = (get_item_key_fc)g_regedit_panel_class_get_key; panel = G_PANEL_ITEM_CLASS(class); panel->dock_at_startup = gtk_panel_item_class_return_false; panel->get_path = (get_panel_path_fc)g_regedit_panel_class_get_path; } /****************************************************************************** * * * Paramètres : panel = instance à initialiser. * * * * Description : Initialise une instance de panneau de paramètres de config. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_regedit_panel_init(GRegeditPanel *panel) { GPanelItem *pitem; /* Version parente du panneau */ GtkBuilder *builder; /* Constructeur utilisé */ GObject *vrenderer; /* Moteur de rendu de colonne */ GtkTreeSortable *sortable; /* Autre vision de la liste */ /* Eléments de base */ pitem = G_PANEL_ITEM(panel); pitem->widget = G_NAMED_WIDGET(gtk_built_named_widget_new_for_panel(_("Configuration"), _("Configuration parameters"), PANEL_REGEDIT_ID)); panel->config = get_main_configuration(); g_object_ref(G_OBJECT(panel->config)); /* Représentation graphique */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(pitem->widget)); vrenderer = G_OBJECT(gtk_builder_get_object(builder, "vrenderer")); g_object_set(vrenderer, "editable", TRUE, NULL); /* Tri de la liste */ sortable = GTK_TREE_SORTABLE(gtk_builder_get_object(builder, "store")); gtk_tree_sortable_set_sort_func(sortable, CPC_PATH, compare_config_list_columns, GINT_TO_POINTER(CPC_PATH), NULL); gtk_tree_sortable_set_sort_func(sortable, CPC_STATUS, compare_config_list_columns, GINT_TO_POINTER(CPC_STATUS), NULL); gtk_tree_sortable_set_sort_func(sortable, CPC_TYPE, compare_config_list_columns, GINT_TO_POINTER(CPC_TYPE), NULL); gtk_tree_sortable_set_sort_func(sortable, CPC_VALUE, compare_config_list_columns, GINT_TO_POINTER(CPC_VALUE), NULL); gtk_tree_sortable_set_sort_column_id(sortable, CPC_PATH, GTK_SORT_ASCENDING); /* Préparation du menu contextuel */ panel->menu = build_param_panel_menu(panel); /* Connexion des signaux */ gtk_builder_add_callback_symbols(builder, BUILDER_CALLBACK(on_param_search_changed), BUILDER_CALLBACK(track_focus_change_in_text_area), BUILDER_CALLBACK(on_button_press_over_params), BUILDER_CALLBACK(on_key_pressed_over_params), BUILDER_CALLBACK(on_param_value_edited), NULL); gtk_builder_connect_signals(builder, panel); g_object_unref(G_OBJECT(builder)); /* Actualisation du contenu du panneau */ reload_config_into_treeview(panel); g_signal_connect(panel->config, "modified", G_CALLBACK(on_configuration_param_modified), panel); } /****************************************************************************** * * * Paramètres : panel = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_regedit_panel_dispose(GRegeditPanel *panel) { if (panel->config != NULL) g_signal_handlers_disconnect_by_func(panel->config, G_CALLBACK(on_configuration_param_modified), panel); g_clear_object(&panel->config); G_OBJECT_CLASS(g_regedit_panel_parent_class)->dispose(G_OBJECT(panel)); } /****************************************************************************** * * * Paramètres : panel = instance d'objet GLib à traiter. * * * * Description : Procède à la libération totale de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_regedit_panel_finalize(GRegeditPanel *panel) { if (panel->filter != NULL) { regfree(panel->filter); free(panel->filter); } G_OBJECT_CLASS(g_regedit_panel_parent_class)->finalize(G_OBJECT(panel)); } /****************************************************************************** * * * Paramètres : class = classe à consulter. * * * * Description : Fournit le nom interne attribué à l'élément réactif. * * * * Retour : Désignation (courte) de l'élément de l'éditeur. * * * * Remarques : - * * * ******************************************************************************/ static char *g_regedit_panel_class_get_key(const GRegeditPanelClass *class) { char *result; /* Description à renvoyer */ result = strdup(PANEL_REGEDIT_ID); return result; } /****************************************************************************** * * * Paramètres : class = classe à consulter. * * * * Description : Indique le chemin initial de la localisation d'un panneau. * * * * Retour : Chemin fixé associé à la position initiale. * * * * Remarques : - * * * ******************************************************************************/ static char *g_regedit_panel_class_get_path(const GRegeditPanelClass *class) { char *result; /* Emplacement à retourner */ result = strdup("M"); return result; } /****************************************************************************** * * * Paramètres : - * * * * Description : Crée un panneau d'affichage des paramètres de configuration. * * * * Retour : Adresse de la structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ GPanelItem *g_regedit_panel_new(void) { GPanelItem *result; /* Structure à retourner */ result = g_object_new(G_TYPE_REGEDIT_PANEL, NULL); return result; } /* ---------------------------------------------------------------------------------- */ /* AFFICHAGE A L'AIDE D'UNE LISTE */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : panel = panneau d'affichage de paramètres de configuration. * * * * Description : Recharge une configuration donnée à l'affichage. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void reload_config_into_treeview(GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkListStore *store; /* Modèle de gestion */ GList *params; /* Paramètres de configuration */ GCfgParam *param; /* Paramètre en cours d'étude */ GList *p; /* Boucle de parcours */ char *type_desc; /* Type de paramètre */ GtkTreeIter iter; /* Point d'insertion */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); store = GTK_LIST_STORE(gtk_builder_get_object(builder, "store")); gtk_list_store_clear(store); g_generic_config_rlock(panel->config); params = g_generic_config_list_params(panel->config); for (p = g_list_first(params); p != NULL; p = g_list_next(p)) { param = G_CFG_PARAM(p->data); if (is_param_filtered(panel, g_config_param_get_path(param))) continue; switch (g_config_param_get_ptype(param)) { case CPT_BOOLEAN: type_desc = _("Boolean"); break; case CPT_INTEGER: type_desc = _("Integer"); break; case CPT_ULONG: type_desc = _("Unsigned long"); break; case CPT_STRING: type_desc = _("String"); break; case CPT_COLOR: type_desc = _("Color"); break; default: type_desc = _("<Unknown type>"); break; } gtk_list_store_append(store, &iter); gtk_list_store_set(store, &iter, CPC_PARAM, param, CPC_PATH, g_config_param_get_path(param), CPC_TYPE, type_desc, -1); update_config_param_value(store, &iter); } g_generic_config_runlock(panel->config); g_object_unref(G_OBJECT(builder)); } /****************************************************************************** * * * Paramètres : config = configuration changée dans son ensemble. * * param = instance dont le contenu a évolué. * * panel = panneau d'affichage de paramètres à mettre à jour. * * * * Description : Actualise l'affichage des données d'un paramètre modifié. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void on_configuration_param_modified(GGenConfig *config, GCfgParam *param, GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkTreeView *treeview; /* Affichage de la liste */ GtkTreeModel *model; /* Gestionnaire de données */ GtkTreeIter iter; /* Point de recherche */ gboolean looping; /* Autorisation de bouclage */ GCfgParam *item; /* Elément de la liste */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview")); model = gtk_tree_view_get_model(treeview); for (looping = gtk_tree_model_get_iter_first(model, &iter); looping; looping = gtk_tree_model_iter_next(model, &iter)) { gtk_tree_model_get(model, &iter, CPC_PARAM, &item, -1); if (item == param) { update_config_param_value(GTK_LIST_STORE(model), &iter); break; } } g_object_unref(G_OBJECT(builder)); } /****************************************************************************** * * * Paramètres : store = gestionnaire du tableau de données. * * iter = point de modification dans les lignes. * * param = paramètre dont la valeur est à afficher. * * * * Description : Actualise la valeur affichée d'un paramètre de configuration.* * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void update_config_param_value(GtkListStore *store, GtkTreeIter *iter) { GCfgParam *param; /* Paramètre à consulter */ ConfigParamState state; /* Etat du paramètre */ char *state_desc; /* Version chaînée de l'état */ bool boolean; /* Valeur booléenne */ int integer; /* Valeur entière */ char int_val[sizeof(XSTR(INT_MIN)) + 1];/* Valeur en chaîne de carac. */ unsigned long ulong; /* Valeur entière positive */ char ul_val[sizeof(XSTR(ULONG_MAX)) + 1];/* Valeur en chaîne de carac. */ char *string; /* Chaîne de caractères */ char *desc; /* Description à afficher */ gtk_tree_model_get(GTK_TREE_MODEL(store), iter, CPC_PARAM, ¶m, -1); state = g_config_param_get_state(param); if (state & CPS_DEFAULT) state_desc = strdup(_("By default")); else state_desc = strdup(_("Changed")); if (state & CPS_EMPTY) state_desc = stradd(state_desc, _(" + empty")); if (state & CPS_EMPTY) desc = ""; else switch (g_config_param_get_ptype(param)) { case CPT_BOOLEAN: g_config_param_get_value(param, &boolean); desc = (boolean ? _("true") : _("false")); break; case CPT_INTEGER: g_config_param_get_value(param, &integer); snprintf(int_val, sizeof(int_val), "%d", integer); desc = int_val; break; case CPT_ULONG: g_config_param_get_value(param, &ulong); snprintf(ul_val, sizeof(ul_val), "%lu", ulong); desc = ul_val; break; case CPT_STRING: g_config_param_get_value(param, &string); desc = (string != NULL ? string : ""); break; case CPT_COLOR: desc = "<color>"; break; default: assert(false); desc = "???"; break; } gtk_list_store_set(store, iter, CPC_BOLD, state & CPS_DEFAULT ? 400 : 800, CPC_STATUS, state_desc, CPC_VALUE, desc, -1); free(state_desc); g_object_unref(G_OBJECT(param)); } /****************************************************************************** * * * Paramètres : model = gestionnaire du tableau de données. * * a = première ligne de données à traiter. * * b = seconde ligne de données à traiter. * * column = indice de la colonne à considérer, encodée. * * * * Description : Etablit une comparaison entre deux lignes de paramètres. * * * * Retour : Indication de tri entre les deux lignes fournies. * * * * Remarques : - * * * ******************************************************************************/ static gint compare_config_list_columns(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer column) { gint result; /* Valeur calculée à retourner */ gchar *value_a; /* Cellule de la ligne 'a' */ gchar *value_b; /* Cellule de la ligne 'b' */ gtk_tree_model_get(model, a, GPOINTER_TO_INT(column), &value_a, -1); gtk_tree_model_get(model, b, GPOINTER_TO_INT(column), &value_b, -1); if (value_a == NULL || value_b == NULL) { if (value_a == NULL && value_b == NULL) result = 0; else result = (value_a == NULL ? -1 : 1); } else result = g_utf8_collate(value_a, value_b); g_free(value_a); g_free(value_b); return result; } /****************************************************************************** * * * Paramètres : treeview = composant graphique présentant les paramètres. * * event = informations liées à l'événement. * * panel = panneau d'affichage sur lequel s'appuyer. * * * * Description : Réagit à une pression sur <Shift+F2> et simule l'édition. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_key_pressed_over_params(GtkTreeView *treeview, GdkEventKey *event, GRegeditPanel *panel) { const gchar *accelerator; /* Combinaison de raccourci */ guint accel_key; /* Touche de raccourci */ GdkModifierType accel_mod; /* Modifiateurs attendus aussi */ GtkTreeIter iter; /* Point de la sélection */ GtkTreeModel *model; /* Gestionnaire de données */ GtkTreePath *path; /* Chemin d'accès à ce point */ if (!g_generic_config_get_value(get_main_configuration(), MPK_KEYBINDINGS_EDIT, &accelerator)) return FALSE; if (accelerator == NULL) return FALSE; gtk_accelerator_parse(accelerator, &accel_key, &accel_mod); if (event->keyval == accel_key && event->state == accel_mod) { if (get_selected_panel_param(treeview, &iter) != NULL) { model = gtk_tree_view_get_model(treeview); path = gtk_tree_model_get_path(model, &iter); gtk_tree_view_set_cursor(treeview, path, gtk_tree_view_get_column(treeview, CPC_VALUE - CPC_PATH), TRUE); gtk_tree_path_free(path); } } return FALSE; } /****************************************************************************** * * * Paramètres : renderer = moteur de rendu pour la cellule. * * path = chemin d'accès vers la cellule éditée. * * new = nouvelle valeur sous forme de texte à valider. * * panel = panneau d'affichage sur lequel s'appuyer. * * * * Description : Réagit à une édition de la valeur d'un paramètre. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void on_param_value_edited(GtkCellRendererText *renderer, gchar *path, gchar *new, GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkListStore *store; /* Modèle de gestion */ GtkTreePath *tree_path; /* Chemin d'accès natif */ GtkTreeIter iter; /* Point de la modification */ GCfgParam *param; /* Paramètre à actualiser */ bool boolean; /* Valeur booléenne */ int integer; /* Valeur entière */ int ulong; /* Valeur entière positive */ char *end; /* Pointeur vers '\0' final ? */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); store = GTK_LIST_STORE(gtk_builder_get_object(builder, "store")); tree_path = gtk_tree_path_new_from_string(path); if (tree_path == NULL) goto bad_path; if (!gtk_tree_model_get_iter(GTK_TREE_MODEL(store), &iter, tree_path)) goto opve_bad_iter; gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, CPC_PARAM, ¶m, -1); switch (g_config_param_get_ptype(param)) { case CPT_BOOLEAN: if (strcmp(new, "true") != 0 && strcmp(new, "false") != 0) goto opve_bad_value; boolean = (strcmp(new, "true") == 0); g_config_param_set_value(param, boolean); break; case CPT_INTEGER: integer = strtol(new, &end, 10); if (*end != '\0') goto opve_bad_value; g_config_param_set_value(param, integer); break; case CPT_ULONG: ulong = strtoul(new, &end, 10); if (*end != '\0') goto opve_bad_value; g_config_param_set_value(param, ulong); break; case CPT_STRING: g_config_param_set_value(param, new); break; case CPT_COLOR: break; default: assert(false); goto opve_bad_value; break; } opve_bad_value: g_object_unref(G_OBJECT(param)); opve_bad_iter: gtk_tree_path_free(tree_path); bad_path: g_object_unref(G_OBJECT(builder)); } /* ---------------------------------------------------------------------------------- */ /* FILTRAGE DES SYMBOLES PRESENTS */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : entry = entrée de texte contenant le filtre brut. * * panel = panneau assurant l'affichage des paramètres. * * * * Description : Démarre l'actualisation du filtrage des paramètres. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void on_param_search_changed(GtkSearchEntry *entry, GRegeditPanel *panel) { const gchar *text; /* Texte de l'utilisateur */ GtkStyleContext *context; /* Contexte du thème actuel */ int ret; /* Bilan de mise en place */ if (panel->filter != NULL) { regfree(panel->filter); free(panel->filter); panel->filter = NULL; } text = gtk_entry_get_text(GTK_ENTRY(entry)); context = gtk_widget_get_style_context(GTK_WIDGET(entry)); if (strlen(text) > 0) { panel->filter = (regex_t *)calloc(1, sizeof(regex_t)); ret = regcomp(panel->filter, text, REG_EXTENDED); if (ret != 0) { free(panel->filter); panel->filter = NULL; gtk_style_context_add_class(context, "filter-error"); return; } } gtk_style_context_remove_class(context, "filter-error"); reload_config_into_treeview(panel); } /****************************************************************************** * * * Paramètres : panel = panneau assurant l'affichage des paramètres. * * name = chemin d'accès au paramètre à traiter. * * * * Description : Détermine si un paramètre doit être filtré ou non. * * * * Retour : true si le paramètre ne doit pas être affiché, false sinon. * * * * Remarques : - * * * ******************************************************************************/ static bool is_param_filtered(GRegeditPanel *panel, const char *name) { regmatch_t match; /* Récupération des trouvailles*/ int ret; /* Bilan du filtrage */ if (panel->filter == NULL) return false; ret = regexec(panel->filter, name, 1, &match, 0); if (ret == REG_NOMATCH) return true; return false; } /* ---------------------------------------------------------------------------------- */ /* ATTRIBUTION D'UN MENU CONTEXTUEL */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : widget = composant GTK visé par l'opération. * * event = informations liées à l'événement. * * panel = informations liées au panneau associé. * * * * Description : Assure la gestion des clics de souris sur les paramètres. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_button_press_over_params(GtkWidget *widget, GdkEventButton *event, GRegeditPanel *panel) { if (event->button == 3) gtk_menu_popup_at_pointer(panel->menu, (GdkEvent *)event); return FALSE; } /****************************************************************************** * * * Paramètres : panel = panneau d'affichage des paramètres de configuration. * * * * Description : Construit le menu contextuel pour les paramètres. * * * * Retour : Panneau de menus mis en place. * * * * Remarques : - * * * ******************************************************************************/ GtkMenu *build_param_panel_menu(GRegeditPanel *panel) { GtkWidget *result; /* Support à retourner */ GtkWidget *submenuitem; /* Sous-élément de menu */ result = qck_create_menu(NULL); submenuitem = qck_create_menu_item(NULL, NULL, _("Copy the name"), G_CALLBACK(mcb_param_panel_copy), panel); gtk_container_add(GTK_CONTAINER(result), submenuitem); submenuitem = qck_create_menu_separator(); gtk_container_add(GTK_CONTAINER(result), submenuitem); submenuitem = qck_create_menu_item(NULL, NULL, _("Make empty"), G_CALLBACK(mcb_param_panel_empty), panel); gtk_container_add(GTK_CONTAINER(result), submenuitem); submenuitem = qck_create_menu_item(NULL, NULL, _("Reset"), G_CALLBACK(mcb_param_panel_reset), panel); gtk_container_add(GTK_CONTAINER(result), submenuitem); return GTK_MENU(result); } /****************************************************************************** * * * Paramètres : treeview = liste d'affichage à consulter. * * save = zone de conservation du point de trouvaille. [OUT]* * * * Description : Fournit le paramètre sélectionné dans la liste. * * * * Retour : Paramètre en cours d'édition ou NULL en cas de soucis. * * * * Remarques : Le résultat non nul est à déréférencer après usage. * * * ******************************************************************************/ static GCfgParam *get_selected_panel_param(GtkTreeView *treeview, GtkTreeIter *save) { GCfgParam *result; /* Paramètre à renvoyer */ GtkTreeSelection *selection; /* Représentation de sélection */ GtkTreeModel *model; /* Gestionnaire des données */ GtkTreeIter iter; /* Point de la sélection */ result = NULL; selection = gtk_tree_view_get_selection(treeview); if (gtk_tree_selection_get_selected(selection, &model, &iter)) gtk_tree_model_get(model, &iter, CPC_PARAM, &result, -1); if (save != NULL) *save = iter; return result; } /****************************************************************************** * * * Paramètres : menuitem = élément de menu sélectionné. * * panel = panneau d'affichage des paramètres de config. * * * * Description : Réagit avec le menu "Copier le nom". * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void mcb_param_panel_copy(GtkMenuItem *menuitem, GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkTreeView *treeview; /* Affichage de la liste */ GCfgParam *param; /* Paramètre sélectionné */ const char *content; /* Prochain contenu à diffuser */ gint clen; /* Taille de ce contenu */ GtkClipboard *clipboard; /* Presse-papiers à remplir */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview")); param = get_selected_panel_param(treeview, NULL); if (param == NULL) { content = g_config_param_get_path(param); clen = g_utf8_strlen(content, -1); clipboard = gtk_clipboard_get(GDK_SELECTION_CLIPBOARD); gtk_clipboard_set_text(clipboard, content, clen); clipboard = gtk_clipboard_get(GDK_SELECTION_PRIMARY); gtk_clipboard_set_text(clipboard, content, clen); g_object_unref(G_OBJECT(param)); } g_object_unref(G_OBJECT(builder)); } /****************************************************************************** * * * Paramètres : menuitem = élément de menu sélectionné. * * panel = panneau d'affichage des paramètres de config. * * * * Description : Réagit avec le menu "Valeur néant". * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void mcb_param_panel_empty(GtkMenuItem *menuitem, GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkTreeView *treeview; /* Affichage de la liste */ GCfgParam *param; /* Paramètre sélectionné */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview")); param = get_selected_panel_param(treeview, NULL); if (param == NULL) { g_config_param_make_empty(param); g_object_unref(G_OBJECT(param)); } g_object_unref(G_OBJECT(builder)); } /****************************************************************************** * * * Paramètres : menuitem = élément de menu sélectionné. * * panel = panneau d'affichage des paramètres de config. * * * * Description : Réagit avec le menu "Réinitialiser". * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void mcb_param_panel_reset(GtkMenuItem *menuitem, GRegeditPanel *panel) { GtkBuilder *builder; /* Constructeur utilisé */ GtkTreeView *treeview; /* Affichage de la liste */ GCfgParam *param; /* Paramètre sélectionné */ builder = gtk_built_named_widget_get_builder(GTK_BUILT_NAMED_WIDGET(G_PANEL_ITEM(panel)->widget)); treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview")); param = get_selected_panel_param(treeview, NULL); if (param == NULL) { g_config_param_reset(param); g_object_unref(G_OBJECT(param)); } g_object_unref(G_OBJECT(builder)); }