/* OpenIDA - Outil d'analyse de fichiers binaires * gtkbufferview.c - affichage de tampons de lignes * * Copyright (C) 2010-2012 Cyrille Bagard * * This file is part of OpenIDA. * * OpenIDA 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. * * OpenIDA 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 Foobar. If not, see . */ #include "gtkbufferview-int.h" /* -------------------------- INTERACTION DIRECTE AVEC GTK -------------------------- */ /* Procède à l'initialisation de l'afficheur de tampons. */ static void gtk_buffer_view_class_init(GtkBufferViewClass *); /* Procède à l'initialisation de l'afficheur de tampons. */ static void gtk_buffer_view_init(GtkBufferView *); /* Fournit la taille de composant requise pour un plein rendu. */ static void gtk_buffer_view_size_request(GtkWidget *, GtkRequisition *); /* S'adapte à la surface concédée par le composant parent. */ static void gtk_buffer_view_size_allocate(GtkWidget *, GtkAllocation *); /* Met à jour l'affichage de la visualisation de code buffer. */ static gboolean gtk_buffer_view_expose(GtkWidget *, GdkEventExpose *); /* Indique la position d'affichage d'une adresse donnée. */ static bool gtk_buffer_view_get_address_coordinates(const GtkBufferView *, vmpa_t, gint *, gint *); /* Réagit à un défilement quelconque. */ static void gtk_buffer_view_scroll(GtkBufferView *); /* Place en cache un rendu destiné à l'aperçu graphique rapide. */ static void gtk_buffer_view_cache_glance(GtkBufferView *, cairo_t *); /* ---------------------------------------------------------------------------------- */ /* INTERACTION DIRECTE AVEC GTK */ /* ---------------------------------------------------------------------------------- */ /* Détermine le type du composant d'affichage de tampon de lignes. */ G_DEFINE_TYPE(GtkBufferView, gtk_buffer_view, GTK_TYPE_VIEW_PANEL) /****************************************************************************** * * * Paramètres : class = classe GTK à initialiser. * * * * Description : Procède à l'initialisation de l'afficheur de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_class_init(GtkBufferViewClass *class) { GtkWidgetClass *widget_class; /* Classe version Widget */ widget_class = GTK_WIDGET_CLASS(class); widget_class->size_request = gtk_buffer_view_size_request; widget_class->size_allocate = gtk_buffer_view_size_allocate; widget_class->expose_event = gtk_buffer_view_expose; } /****************************************************************************** * * * Paramètres : view = composant GTK à initialiser. * * * * Description : Procède à l'initialisation de l'afficheur de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_init(GtkBufferView *view) { GtkViewPanel *viewpanel; /* Instance parente */ viewpanel = GTK_VIEW_PANEL(view); viewpanel->get_coordinates = (get_addr_coordinates_fc)gtk_buffer_view_get_address_coordinates; viewpanel->scroll = (scroll_fc)gtk_buffer_view_scroll; viewpanel->cache_glance = (cache_glance_fc)gtk_buffer_view_cache_glance; } /****************************************************************************** * * * Paramètres : view = composant GTK à consulter. * * event = informations liées à l'événement. * * * * Description : Transcrit les coordonnées à l'écran en coordonnées absolues. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_buffer_view_compute_fake_coord(GtkBufferView *view, gint *x, gint *y) { if (GTK_VIEW_PANEL(view)->hadjustment != NULL) *x -= gtk_adjustment_get_value(GTK_VIEW_PANEL(view)->hadjustment); if (GTK_VIEW_PANEL(view)->vadjustment != NULL) *y += gtk_adjustment_get_value(GTK_VIEW_PANEL(view)->vadjustment); } /****************************************************************************** * * * Paramètres : view = composant GTK à consulter. * * event = informations liées à l'événement. * * * * Description : Transcrit les coordonnées absolues en coordonnées à l'écran. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_buffer_view_compute_real_coord(GtkBufferView *view, gint *x, gint *y) { if (x != NULL && GTK_VIEW_PANEL(view)->hadjustment != NULL) *x += gtk_adjustment_get_value(GTK_VIEW_PANEL(view)->hadjustment); if (y != NULL && GTK_VIEW_PANEL(view)->vadjustment != NULL) *y += gtk_adjustment_get_value(GTK_VIEW_PANEL(view)->vadjustment); } /****************************************************************************** * * * Paramètres : widget = composant GTK à consulter. * * requisition = dimensions souhaitées. [OUT] * * * * Description : Fournit la taille de composant requise pour un plein rendu. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_size_request(GtkWidget *widget, GtkRequisition *requisition) { GtkBufferView *view; /* Autre version du composant */ view = GTK_BUFFER_VIEW(widget); if (view->buffer_view != NULL) g_buffer_view_get_size(view->buffer_view, &requisition->width, &requisition->height, *GTK_VIEW_PANEL(view)->display_addr, *GTK_VIEW_PANEL(view)->display_code); } /****************************************************************************** * * * Paramètres : view = composant GTK à mettre à jour. * * allocation = étendue accordée à la vue. * * * * Description : S'adapte à la surface concédée par le composant parent. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_size_allocate(GtkWidget *widget, GtkAllocation *allocation) { GtkViewPanel *panel; /* Autre version du composant */ GtkBufferView *view; /* Encore une autre version */ gint width; /* Largeur de l'objet actuelle */ gint height; /* Hauteur de l'objet actuelle */ GtkAllocation valloc; /* Surface utilisable */ gboolean changed; /* Changement de valeur ? */ /* Mise à jour GTK */ widget->allocation = *allocation; if (gtk_widget_get_realized(widget)) gdk_window_move_resize(widget->window, allocation->x, allocation->y, allocation->width, allocation->height); panel = GTK_VIEW_PANEL(widget); if (panel->hadjustment == NULL || panel->vadjustment == NULL) return; view = GTK_BUFFER_VIEW(widget); g_buffer_view_get_size(view->buffer_view, &width, &height, *panel->display_addr, *panel->display_code); gtk_view_panel_compute_allocation(panel, &valloc); /* Défilement horizontal */ panel->hadjustment->page_size = valloc.width; panel->hadjustment->step_increment = valloc.width * 0.1; panel->hadjustment->page_increment = valloc.width * 0.9; panel->hadjustment->upper = MAX(width, valloc.width); gtk_view_panel_reclamp_adjustment(panel->hadjustment, &changed); gtk_adjustment_changed(panel->hadjustment); if (changed) gtk_adjustment_value_changed(panel->hadjustment); /* Défilement vertical */ panel->vadjustment->page_size = valloc.height; panel->vadjustment->step_increment = view->line_height; panel->vadjustment->page_increment = panel->vadjustment->step_increment * 10.0; panel->vadjustment->upper = MAX(height, valloc.height); gtk_view_panel_reclamp_adjustment(panel->vadjustment, &changed); gtk_adjustment_changed(panel->vadjustment); if (changed) gtk_adjustment_value_changed(panel->vadjustment); } /****************************************************************************** * * * Paramètres : view = composant GTK à redessiner. * * event = informations liées à l'événement. * * * * Description : Met à jour l'affichage de la visualisation de code buffer. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean gtk_buffer_view_expose(GtkWidget *widget, GdkEventExpose *event) { GtkBufferView *view; /* Autre version du composant */ GtkViewPanel *pview; /* Autre version du composant */ GtkStyle *style; /* Style associé au composant */ GdkDrawable *drawable; /* Surface de dessin */ gint fake_x; /* Abscisse virtuelle */ gint fake_y; /* Ordonnée virtuelle */ GtkStateType state; /* Etat du composant */ GtkViewPanelClass *parent_class; /* Version pure du parent */ view = GTK_BUFFER_VIEW(widget); widget = GTK_WIDGET(view); pview = GTK_VIEW_PANEL(widget); drawable = GDK_DRAWABLE(event->window); gdk_window_begin_paint_region(drawable, event->region); gdk_gc_set_clip_region(pview->gc, event->region); style = gtk_widget_get_style(GTK_WIDGET(view)); fake_x = 0; fake_y = 0; gtk_buffer_view_compute_fake_coord(view, &fake_x, &fake_y); /* Dessin de la marge gauche */ state = gtk_widget_get_state(widget); gdk_gc_set_foreground(pview->gc, &style->mid[state]); gdk_draw_rectangle(drawable, pview->gc, TRUE, fake_x, event->area.y, view->left_margin, event->area.y + event->area.height); gdk_gc_set_foreground(pview->gc, &style->dark[state]); gdk_draw_line(drawable, pview->gc, fake_x + view->left_margin, event->area.y, fake_x + view->left_margin, event->area.y + event->area.height); /* Eventuelle bordure globale */ parent_class = g_type_class_peek(g_type_parent(GTK_TYPE_BUFFER_VIEW)); GTK_WIDGET_CLASS(parent_class)->expose_event(widget, event); /* Impression du désassemblage */ if (view->buffer_view != NULL) g_buffer_view_draw(view->buffer_view, event, pview->gc, fake_x, fake_y, *pview->display_addr, *pview->display_code); gdk_window_end_paint(drawable); return TRUE; } /****************************************************************************** * * * Paramètres : view = composant GTK à consulter. * * addr = adresse à présenter à l'écran. * * x = position horizontale au sein du composant. [OUT] * * y = position verticale au sein du composant. [OUT] * * * * Description : Indique la position d'affichage d'une adresse donnée. * * * * Retour : true si l'adresse fait partie du composant, false sinon. * * * * Remarques : - * * * ******************************************************************************/ static bool gtk_buffer_view_get_address_coordinates(const GtkBufferView *view, vmpa_t addr, gint *x, gint *y) { return g_buffer_view_get_address_coordinates(view->buffer_view, addr, x, y); } /****************************************************************************** * * * Paramètres : view = composant GTK à mettre à jour. * * * * Description : Réagit à un défilement quelconque. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_scroll(GtkBufferView *view) { gtk_widget_queue_draw(GTK_WIDGET(view)); } /****************************************************************************** * * * Paramètres : view = composant GTK à manipuler. * * cairo = assistant pour la création de rendus. * * * * Description : Place en cache un rendu destiné à l'aperçu graphique rapide. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_cache_glance(GtkBufferView *view, cairo_t *cairo) { GtkRequisition req; /* Pleine taille de la source */ gtk_widget_size_request(GTK_WIDGET(view), &req); gtk_buffer_view_compute_real_coord(view, &req.width, &req.height); /* FIXME : hardcoded 20 */ cairo_set_line_width(cairo, 20); cairo_set_source_rgb(cairo, 0, 0, 0); cairo_rectangle(cairo, 0, 0, req.width - 20, req.height - 20); cairo_stroke(cairo); } /****************************************************************************** * * * Paramètres : view = composant GTK à mettre à jour. * * buffer = tampon de lignes à encadrer. * * addr = indique si les positions doivent être affichées. * * code = indique si le code binaire doit être affiché. * * * * Description : Prend acte de l'association d'un tampon de lignes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_buffer_view_attach_buffer(GtkBufferView *view, GBufferView *buffer, bool *addr, bool *code) { gint width; /* Largeur de l'objet actuelle */ gint height; /* Hauteur de l'objet actuelle */ if (view->buffer != NULL) { g_object_unref(G_OBJECT(view->buffer)); g_object_unref(G_OBJECT(view->buffer_view)); } view->buffer = g_buffer_view_get_buffer(buffer); g_object_ref(G_OBJECT(view->buffer)); view->buffer_view = buffer; //gdk_threads_enter(); /* Taille des marges */ view->line_height = g_buffer_view_get_line_height(view->buffer_view); view->left_margin = 2 * view->line_height; view->left_text = -2.5 * view->line_height; /* Validation finale */ g_buffer_view_get_size(view->buffer_view, &width, &height, *addr, *code); width += -view->left_text + 1; height += 1; //gtk_widget_set_size_request(GTK_WIDGET(view), width, height); gtk_widget_queue_draw(GTK_WIDGET(view)); //gdk_flush (); //gdk_threads_leave(); } /****************************************************************************** * * * Paramètres : view = composant GTK à consulter. * * * * Description : Fournit la vue associée au tampon de lignes courant. * * * * Retour : Vue mise en place. * * * * Remarques : - * * * ******************************************************************************/ GBufferView *gtk_buffer_view_get_buffer(const GtkBufferView *view) { return view->buffer_view; }