/* Chrysalide - Outil d'analyse de fichiers binaires * log.c - panneau d'affichage des messages système * * Copyright (C) 2009-2013 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 "log.h" #include #include #include #include #include #include "panel-int.h" #include "../core/panels.h" /* Colonnes de la liste des messages */ typedef enum _LogColumn { LGC_PICTURE, /* Image de représentation */ LGC_STRING, /* Chaîne de caractères */ LGC_COUNT /* Nombre de colonnes */ } LogColumn; /* Paramètres à transmettre pour un affichage */ typedef struct _log_data { GPanelItem *item; /* Intermédiaire mis en place */ LogMessageType type; /* Type de message à afficher */ char *msg; /* Contenu du message */ } log_data; /* Paramètres à transmettre pour un défilement */ typedef struct _scroll_data { GtkTreeView *treeview; /* Affichage de la liste */ GtkTreePath *path; /* Chemin d'accès à cibler */ } scroll_data; /* Panneau d'accueil (instance) */ struct _GLogPanel { GPanelItem parent; /* A laisser en premier */ }; /* Panneau d'accueil (classe) */ struct _GLogPanelClass { GPanelItemClass parent; /* A laisser en premier */ }; /* Initialise la classe des panneaux d'affichage des messages. */ static void g_log_panel_class_init(GLogPanelClass *); /* Initialise une instance de panneau d'affichage des messages. */ static void g_log_panel_init(GLogPanel *); /* Supprime toutes les références externes. */ static void g_log_panel_dispose(GLogPanel *); /* Procède à la libération totale de la mémoire. */ static void g_log_panel_finalize(GLogPanel *); /* Affiche un message dans le journal des messages système. */ static void _log_simple_message(LogMessageType, char *); /* Affiche un message dans le journal des messages système. */ static gboolean log_message(log_data *); /* Indique le type définit pour un panneau d'affichage de messages. */ G_DEFINE_TYPE(GLogPanel, g_log_panel, G_TYPE_PANEL_ITEM); /****************************************************************************** * * * Paramètres : klass = classe à initialiser. * * * * Description : Initialise la classe des panneaux d'affichage des messages. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_log_panel_class_init(GLogPanelClass *klass) { GObjectClass *object; /* Autre version de la classe */ GPanelItemClass *panel; /* Version parente de la classe*/ object = G_OBJECT_CLASS(klass); object->dispose = (GObjectFinalizeFunc/* ! */)g_log_panel_dispose; object->finalize = (GObjectFinalizeFunc)g_log_panel_finalize; panel = G_PANEL_ITEM_CLASS(klass); panel->unique = true; panel->bindings = "F1"; } /****************************************************************************** * * * Paramètres : panel = instance à initialiser. * * * * Description : Initialise une instance de panneau d'affichage des messages. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_log_panel_init(GLogPanel *panel) { GEditorItem *base; /* Version basique d'instance */ GPanelItem *pitem; /* Version parente du panneau */ GtkWidget *scrolled; /* Fenêtre avec défilements */ GtkTreeStore *store; /* Modèle de gestion */ GtkWidget *treeview; /* Affichage de la liste */ GtkCellRenderer *renderer; /* Moteur de rendu de colonne */ GtkTreeViewColumn *column; /* Colonne de la liste */ /* Eléments de base */ base = G_EDITOR_ITEM(panel); base->name = PANEL_LOG_ID; pitem = G_PANEL_ITEM(panel); pitem->personality = PIP_SINGLETON; pitem->lname = _("Misc information"); pitem->dock_at_startup = true; pitem->path = strdup("S"); /* Représentation graphique */ scrolled = gtk_scrolled_window_new(NULL, NULL); gtk_widget_show(scrolled); base->widget = scrolled; /* Construction graphique */ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_IN); store = gtk_tree_store_new(LGC_COUNT, G_TYPE_STRING, G_TYPE_STRING); g_object_set_data(G_OBJECT(scrolled), "store", store); treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)); g_object_set_data(G_OBJECT(scrolled), "treeview", treeview); gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE); gtk_widget_show(treeview); gtk_container_add(GTK_CONTAINER(scrolled), treeview); g_object_unref(G_OBJECT(store)); column = gtk_tree_view_column_new(); gtk_tree_view_column_set_visible(column, FALSE); gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); gtk_tree_view_set_expander_column(GTK_TREE_VIEW(treeview), column); column = gtk_tree_view_column_new(); renderer = gtk_cell_renderer_pixbuf_new(); gtk_tree_view_column_pack_start(column, renderer, FALSE); gtk_tree_view_column_add_attribute(column, renderer, "stock-id", LGC_PICTURE); renderer = gtk_cell_renderer_text_new(); gtk_tree_view_column_pack_start(column, renderer, TRUE); gtk_tree_view_column_add_attribute(column, renderer, "markup", LGC_STRING); gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); } /****************************************************************************** * * * Paramètres : panel = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_log_panel_dispose(GLogPanel *panel) { G_OBJECT_CLASS(g_log_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_log_panel_finalize(GLogPanel *panel) { G_OBJECT_CLASS(g_log_panel_parent_class)->finalize(G_OBJECT(panel)); } /****************************************************************************** * * * Paramètres : - * * * * Description : Crée un panneau d'affichage des messages système. * * * * Retour : Adresse de la structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ GPanelItem *g_log_panel_new(void) { GPanelItem *result; /* Structure à retourner */ result = g_object_new(G_TYPE_LOG_PANEL, NULL); return result; } /****************************************************************************** * * * Paramètres : type = espèce du message à ajouter. * * msg = message à faire apparaître à l'écran. * * * * Description : Affiche un message dans le journal des messages système. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void log_simple_message(LogMessageType type, const char *msg) { _log_simple_message(type, strdup(msg)); } /****************************************************************************** * * * Paramètres : type = espèce du message à ajouter. * * msg = message à faire apparaître à l'écran. * * * * Description : Affiche un message dans le journal des messages système. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void _log_simple_message(LogMessageType type, char *msg) { GPanelItem *item; /* Intermédiaire mis en place */ log_data *data; /* Paramètres à joindre */ item = get_panel_item_by_name(PANEL_LOG_ID); data = (log_data *)calloc(1, sizeof(log_data)); data->item = item; data->type = type; data->msg = msg; g_object_ref(G_OBJECT(data->item)); g_main_context_invoke(NULL, (GSourceFunc)log_message, data); } /****************************************************************************** * * * Paramètres : type = espèce du message à ajouter. * * fmt = format du message à faire apparaître à l'écran. * * ... = éventuels arguments venant compléter le message. * * * * Description : Affiche un message dans le journal des messages système. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void log_variadic_message(LogMessageType type, const char *fmt, ...) { size_t len; /* Taille tampon disponible */ char *buffer; /* Tampon du msg reconstitué */ int ret; /* Bilan d'une impression */ char *ptr; /* Nouvelle allocation */ va_list ap; /* Liste d'arguments variable */ len = VARIADIC_LOG_BUFSIZE; buffer = calloc(len, sizeof(char)); while (buffer != NULL) { va_start(ap, fmt); ret = vsnprintf(buffer, len, fmt, ap); va_end(ap); if (ret >= 0 && ret < len) break; else { if (ret > -1) len += 1; /* glibc 2.1 */ else len *= 2; /* glibc 2.0 */ if ((ptr = realloc(buffer, len)) == NULL) { free(buffer); buffer = NULL; } else buffer = ptr; } } _log_simple_message(type, buffer); } /****************************************************************************** * * * Paramètres : data = paramètres destinés à l'affichage d'un message. * * * * Description : Affiche un message dans le journal des messages système. * * * * Retour : - * * * * Remarques : Cette fonction, et c'est tout son intérêt, est toujours * * exécutée dans le contexte GTK principal. * * * ******************************************************************************/ static gboolean log_message(log_data *data) { GtkWidget *panel; /* Panneau à traiter */ GtkTreeStore *store; /* Modèle de gestion */ GtkTreeIter iter; /* Point d'insertion */ GtkTreeView *treeview; /* Affichage de la liste */ GtkTreePath *path; /* Chemin d'accès à la ligne */ scroll_data *sdata; /* Paramètres de défilement */ /* Mise en place du message */ panel = g_editor_item_get_widget(G_EDITOR_ITEM(data->item)); store = g_object_get_data(G_OBJECT(panel), "store"); gtk_tree_store_append(store, &iter, NULL); switch (data->type) { case LMT_INFO: gtk_tree_store_set(store, &iter, LGC_PICTURE, "gtk-info", LGC_STRING, data->msg, -1); break; case LMT_BAD_BINARY: gtk_tree_store_set(store, &iter, LGC_PICTURE, "gtk-dialog-warning", LGC_STRING, data->msg, -1); break; case LMT_PROCESS: gtk_tree_store_set(store, &iter, LGC_PICTURE, "gtk-execute", LGC_STRING, data->msg, -1); break; case LMT_ERROR: gtk_tree_store_set(store, &iter, LGC_PICTURE, "gtk-dialog-error", LGC_STRING, data->msg, -1); break; case LMT_WARNING: gtk_tree_store_set(store, &iter, LGC_PICTURE, "gtk-dialog-warning", LGC_STRING, data->msg, -1); break; default: gtk_tree_store_set(store, &iter, LGC_STRING, data->msg, -1); break; } /* Défilement pour pointer à l'affichage */ gboolean wait_for_scrolling(scroll_data *_data) { gtk_tree_view_scroll_to_cell(_data->treeview, _data->path, NULL, FALSE, 0.0, 0.0); g_object_unref(G_OBJECT(_data->treeview)); gtk_tree_path_free(_data->path); free(_data); return G_SOURCE_REMOVE; } treeview = GTK_TREE_VIEW(g_object_get_data(G_OBJECT(panel), "treeview")); path = gtk_tree_model_get_path(GTK_TREE_MODEL(store), &iter); sdata = (scroll_data *)calloc(1, sizeof(log_data)); sdata->treeview = treeview; sdata->path = path; g_object_ref(G_OBJECT(sdata->treeview)); g_idle_add((GSourceFunc)wait_for_scrolling, sdata); /* Nettoyage de la mémoire */ g_object_unref(G_OBJECT(data->item)); free(data->msg); free(data); return G_SOURCE_REMOVE; }