/* Chrysalide - Outil d'analyse de fichiers binaires * bufferview.c - affichage de tampons de lignes * * Copyright (C) 2016-2024 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>. */ #include "bufferview.h" #include "bufferview-int.h" /* ------------------------- BASES D'UN COMPOSANT GRAPHIQUE ------------------------- */ /* Initialise la classe des afficheurs de tampons. */ static void gtk_buffer_view_class_init(GtkBufferViewClass *); /* Initialise une instance d'afficheur de tampons. */ static void gtk_buffer_view_init(GtkBufferView *); /* Supprime toutes les références externes. */ static void gtk_buffer_view_dispose(GtkBufferView *); /* Procède à la libération totale de la mémoire. */ static void gtk_buffer_view_finalize(GtkBufferView *); /* --------------------- IMPLEMENTATION DES FONCTIONS DE CLASSE --------------------- */ /* Prend acte de la taille allouée au composant d'affichage. */ static void gtk_buffer_view_size_allocate(GtkWidget *, int, int, int); /* Réagit à un défilement chez une barre associée au composant. */ static void gtk_buffer_view_adjust_scroll_value(GtkBufferView *, GtkOrientation, GtkAdjustment *); /* ---------------------------------------------------------------------------------- */ /* BASES D'UN COMPOSANT GRAPHIQUE */ /* ---------------------------------------------------------------------------------- */ /* Détermine le type du composant d'affichage générique. */ G_DEFINE_TYPE(GtkBufferView, gtk_buffer_view, GTK_TYPE_CONTENT_VIEW); /****************************************************************************** * * * Paramètres : class = classe GTK à initialiser. * * * * Description : Initialise la classe des afficheurs de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_class_init(GtkBufferViewClass *class) { GObjectClass *object; /* Autre version de la classe */ GtkWidgetClass *widget; /* Classe version Widget */ GtkContentViewClass *content; /* Base d'affichage */ object = G_OBJECT_CLASS(class); object->dispose = (GObjectFinalizeFunc/* ! */)gtk_buffer_view_dispose; object->finalize = (GObjectFinalizeFunc)gtk_buffer_view_finalize; widget = GTK_WIDGET_CLASS(class); widget->size_allocate = gtk_buffer_view_size_allocate; content = GTK_CONTENT_VIEW_CLASS(class); content->adjust = (adjust_scroll_value_fc)gtk_buffer_view_adjust_scroll_value; #if 0 widget_class->focus_in_event = gtk_buffer_view_focus; widget_class->focus_out_event = gtk_buffer_view_focus; widget_class->button_press_event = gtk_buffer_view_button_press; widget_class->draw = gtk_buffer_view_draw; widget_class->key_press_event = gtk_buffer_view_key_press; panel_class = GTK_DISPLAY_PANEL_CLASS(class); panel_class->compute_size = (compute_requested_size_fc)gtk_buffer_view_compute_requested_size; panel_class->compute_inc = (compute_scroll_inc_fc)gtk_buffer_view_compute_scroll_inc; panel_class->adjust = (adjust_scroll_value_fc)gtk_buffer_view_adjust_scroll_value; panel_class->get_coordinates = (get_coordinates_fc)gtk_buffer_view_get_cursor_coordinates; panel_class->get_active = (get_active_object_fc)gtk_buffer_view_get_active_object; panel_class->move_caret_to = (move_caret_to_fc)_gtk_buffer_view_move_caret_to; panel_class->cache_glance = (cache_glance_fc)gtk_buffer_view_cache_glance; panel_class->get_cursor = (get_cursor_fc)gtk_buffer_view_get_cursor; /* Signaux */ g_signal_new("reach-limit", GTK_TYPE_BUFFER_DISPLAY, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GtkBufferViewClass, reach_limit), NULL, NULL, g_cclosure_marshal_VOID__ENUM, G_TYPE_NONE, 1, GTK_TYPE_SCROLL_TYPE); g_signal_new("prepare-collapsing", GTK_TYPE_BUFFER_DISPLAY, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GtkBufferViewClass, prepare_collapsing), NULL, NULL, g_cclosure_marshal_VOID__BOOLEAN, G_TYPE_NONE, 1, G_TYPE_BOOLEAN); #endif } /****************************************************************************** * * * Paramètres : view = composant GTK à initialiser. * * * * Description : Initialise une instance d'afficheur de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_init(GtkBufferView *view) { view->view = NULL; view->style = g_token_style_new(GTK_WIDGET(view)); view->virt_top = 0; } /****************************************************************************** * * * Paramètres : view = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_dispose(GtkBufferView *view) { #if 0 if (display->caret_timer != 0) { g_source_remove(display->caret_timer); display->caret_timer = 0; } g_clear_object(&display->cursor); g_clear_object(&display->builder); g_clear_object(&display->bar); #endif g_clear_object(&view->view); g_clear_object(&view->style); G_OBJECT_CLASS(gtk_buffer_view_parent_class)->dispose(G_OBJECT(view)); } /****************************************************************************** * * * Paramètres : view = instance d'objet Gtk à traiter. * * * * Description : Procède à la libération totale de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_finalize(GtkBufferView *view) { G_OBJECT_CLASS(gtk_buffer_view_parent_class)->finalize(G_OBJECT(view)); } /****************************************************************************** * * * 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_view(const GtkBufferView *view) { GBufferView *result; /* Instance à retourner */ result = view->view; ref_object(result); return result; } /* ---------------------------------------------------------------------------------- */ /* IMPLEMENTATION DES FONCTIONS DE CLASSE */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : widget = composant GTK à examiner. * * width = largeur affectée au composant graphique. * * height = hauteur affectée au composant graphique. * * baseline = ligne de base affectée au composant graphique. * * * * Description : Prend acte de la taille allouée au composant d'affichage. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_size_allocate(GtkWidget *widget, int width, int height, int baseline) { GtkContentView *base; /* Version de base */ GtkBufferView *view; /* Version spécialisée */ int full_width; /* Largeur idéale complète */ int full_height; /* Hauteur idéale complète */ int *min_widths; /* Largeurs minimales imposées */ int *min_heights; /* Hauteurs minimales imposées */ size_t i; /* Boucle de parcours */ int upper; /* Valeur maximale retenue */ GtkAdjustment *adj; /* Ajustement à mettre à jour */ int line_height; /* Hauteur d'une ligne unique */ int lines_per_page; /* Nombre de lignes sur l'écran*/ GtkAllocation allocated; /* Zone allouée */ GWidthTracker *tracker; /* Collecteur de largeurs */ base = GTK_CONTENT_VIEW(widget); view = GTK_BUFFER_VIEW(widget); /* Détermination de la surface idéale à afficher */ g_buffer_view_compute_size(view->view, &full_width, &full_height); min_widths = alloca(base->sub_count * sizeof(int)); min_heights = alloca(base->sub_count * sizeof(int)); for (i = 0; i < base->sub_count; i++) { gtk_widget_measure(base->sub_children[i], GTK_ORIENTATION_HORIZONTAL, -1, &min_widths[i], NULL, NULL, NULL); full_width += min_widths[i]; gtk_widget_measure(base->sub_children[i], GTK_ORIENTATION_VERTICAL, -1, &min_heights[i], NULL, NULL, NULL); full_height += min_heights[i]; } /* Définition des besoins en défilement */ upper = MAX(width, full_width); adj = base->adjustments[GTK_ORIENTATION_HORIZONTAL]; gtk_adjustment_configure(adj, gtk_adjustment_get_value(adj), 0, upper, 0.1 * width, 0.9 * width, width); upper = MAX(height, full_height); line_height = g_token_style_get_line_height(view->style); lines_per_page = height / line_height; if (lines_per_page == 0) lines_per_page = 1; adj = base->adjustments[GTK_ORIENTATION_VERTICAL]; gtk_adjustment_configure(adj, gtk_adjustment_get_value(adj), 0, upper, line_height * lines_per_page, line_height * lines_per_page * 10, height); /* Placement des composants d'affichage */ tracker = g_buffer_view_get_tracker(view->view); allocated.x = 0; allocated.y = 0; allocated.height = height; for (i = 0; i < base->sub_count; i++) { allocated.width = g_width_tracker_get_column_min_width(tracker, i); if ((i + 1) == base->sub_count && allocated.width < (width - allocated.x)) allocated.width = width - allocated.x; gtk_widget_size_allocate(base->sub_children[i], &allocated, baseline); allocated.x += allocated.width; } unref_object(tracker); /* Mise à jour des éléments plus internes */ GTK_WIDGET_CLASS(gtk_buffer_view_parent_class)->size_allocate(widget, width, height, baseline); } /****************************************************************************** * * * Paramètres : view = panneau d'affichage concerné. * * orientation = indication sur le défilement à traiter. * * adj = nouvel ajustement à prendre en compte. * * * * Description : Réagit à un défilement chez une barre associée au composant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_view_adjust_scroll_value(GtkBufferView *view, GtkOrientation orientation, GtkAdjustment *adj) { GtkContentView *base; /* Version de base */ size_t i; /* Boucle de parcours */ base = GTK_CONTENT_VIEW(view); if (orientation == GTK_ORIENTATION_HORIZONTAL) printf("TODO...\n"); else { view->virt_top = gtk_adjustment_get_value(adj); for (i = 0; i < base->sub_count; i++) gtk_widget_queue_draw(base->sub_children[i]); } } #if 0 #include <assert.h> #include "easygtk.h" #include "gtkbufferdisplay-int.h" #include "../core/params.h" #include "../glibext/gbinarycursor.h" // REMME /* -------------------------- INTERACTION DIRECTE AVEC GTK -------------------------- */ /* Procède à l'initialisation de l'afficheur de tampons. */ //static void gtk_buffer_display_class_init(GtkBufferDisplayClass *); /* Procède à l'initialisation de l'afficheur de tampons. */ //static void gtk_buffer_display_init(GtkBufferDisplay *); /* Supprime toutes les références externes. */ //static void gtk_buffer_display_dispose(GtkBufferDisplay *); /* Procède à la libération totale de la mémoire. */ //static void gtk_buffer_display_finalize(GtkBufferDisplay *); /* Intègre le focus dans le rendu du composant. */ static gboolean gtk_buffer_display_focus(GtkWidget *, GdkEventFocus *); /* Assure la gestion des clics de souris sur le composant. */ static gboolean gtk_buffer_display_button_press(GtkWidget *, GdkEventButton *); /* Met à jour l'affichage de la visualisation de code buffer. */ static gboolean gtk_buffer_display_draw(GtkWidget *, cairo_t *); /* Prend en compte une frappe de touche sur le composant. */ static gboolean gtk_buffer_display_key_press(GtkWidget *, GdkEventKey *); /* Indique les dimensions de travail du composant d'affichage. */ static void gtk_buffer_display_compute_requested_size(GtkBufferDisplay *, gint *, gint *); /* Détermine la taille des bonds lors de défilements. */ static void gtk_buffer_display_compute_scroll_inc(GtkBufferDisplay *, gint, GtkOrientation, gdouble *, gdouble *); /* Réagit à un défilement chez une barre associée au composant. */ static void gtk_buffer_display_adjust_scroll_value(GtkBufferDisplay *, GtkAdjustment *, GtkOrientation); /* Indique la position d'affichage d'un emplacement donné. */ static bool gtk_buffer_display_get_cursor_coordinates(const GtkBufferDisplay *, const GLineCursor *, gint *, gint *, ScrollPositionTweak); /* Fournit l'élément actif lié à la position courante. */ GObject *gtk_buffer_display_get_active_object(const GtkBufferDisplay *); /* Place en cache un rendu destiné à l'aperçu graphique rapide. */ static void gtk_buffer_display_cache_glance(GtkBufferDisplay *, cairo_t *, const GtkAllocation *, double); /* Fournit le position courante dans un panneau de chargement. */ static GLineCursor *gtk_buffer_display_get_cursor(const GtkBufferDisplay *); /* ------------------------------ ANIMATION DU CURSEUR ------------------------------ */ /* Déplace le curseur à un emplacement défini. */ static bool _gtk_buffer_display_move_caret_to(GtkBufferDisplay *, gint, gint); /* Déplace le curseur en effaçant son éventuelle position. */ static void gtk_buffer_display_relocate_caret(GtkBufferDisplay *, const cairo_rectangle_int_t *, GLineCursor *); /* Assure le clignotement du curseur à l'emplacement courant. */ static gboolean gtk_buffer_display_refresh_caret(GtkBufferDisplay *); /* Redémarre l'affichage du curseur à l'emplacement courant. */ static void gtk_buffer_display_restart_caret_blinking(GtkBufferDisplay *); /* Prépare l'actualisation de tout ou une partie de l'affichage. */ static void gtk_buffer_display_queue_draw_caret(GtkBufferDisplay *, cairo_rectangle_int_t *); /* Affiche le curseur à l'écran, s'il doit l'être. */ static void gtk_buffer_display_draw_caret(GtkBufferDisplay *, cairo_t *); /* ------------------------- INCLUSION D'UNE BARRE D'OUTILS ------------------------- */ /* Place correctement la barre d'outils pour bloc. */ static void gtk_buffer_display_move_block_bar(GtkBufferDisplay *); /* Accompagne le début du survol d'un élément de barre d'outils. */ static gboolean on_block_bar_enter_notify(GtkWidget *, GdkEventCrossing *, GtkBufferDisplay *); /* Accompagne la fin du survol d'un élément de barre d'outils. */ static gboolean on_block_bar_leave_notify(GtkWidget *, GdkEventCrossing *, GtkBufferDisplay *); /* Accompagne le début du survol du bouton de compression. */ static gboolean on_block_bar_collapsing_enter(GtkWidget *, GdkEventCrossing *, GtkBufferDisplay *); /* Accompagne la fin du survol du bouton de compression. */ static gboolean on_block_bar_collapsing_leave(GtkWidget *, GdkEventCrossing *, GtkBufferDisplay *); /* ---------------------------------------------------------------------------------- */ /* INTERACTION DIRECTE AVEC GTK */ /* ---------------------------------------------------------------------------------- */ /* Détermine le type du composant d'affichage de tampon de lignes. */ G_DEFINE_TYPE(GtkBufferDisplay, gtk_buffer_display, GTK_TYPE_DISPLAY_PANEL) /****************************************************************************** * * * Paramètres : class = classe GTK à initialiser. * * * * Description : Procède à l'initialisation de l'afficheur de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_class_init(GtkBufferDisplayClass *class) { GObjectClass *object; /* Autre version de la classe */ GtkWidgetClass *widget_class; /* Classe version Widget */ GtkDisplayPanelClass *panel_class; /* Classe parente */ object = G_OBJECT_CLASS(class); object->dispose = (GObjectFinalizeFunc/* ! */)gtk_buffer_display_dispose; object->finalize = (GObjectFinalizeFunc)gtk_buffer_display_finalize; widget_class = GTK_WIDGET_CLASS(class); widget_class->focus_in_event = gtk_buffer_display_focus; widget_class->focus_out_event = gtk_buffer_display_focus; widget_class->button_press_event = gtk_buffer_display_button_press; widget_class->draw = gtk_buffer_display_draw; widget_class->key_press_event = gtk_buffer_display_key_press; panel_class = GTK_DISPLAY_PANEL_CLASS(class); panel_class->compute_size = (compute_requested_size_fc)gtk_buffer_display_compute_requested_size; panel_class->compute_inc = (compute_scroll_inc_fc)gtk_buffer_display_compute_scroll_inc; panel_class->adjust = (adjust_scroll_value_fc)gtk_buffer_display_adjust_scroll_value; panel_class->get_coordinates = (get_coordinates_fc)gtk_buffer_display_get_cursor_coordinates; panel_class->get_active = (get_active_object_fc)gtk_buffer_display_get_active_object; panel_class->move_caret_to = (move_caret_to_fc)_gtk_buffer_display_move_caret_to; panel_class->cache_glance = (cache_glance_fc)gtk_buffer_display_cache_glance; panel_class->get_cursor = (get_cursor_fc)gtk_buffer_display_get_cursor; /* Signaux */ g_signal_new("reach-limit", GTK_TYPE_BUFFER_DISPLAY, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GtkBufferDisplayClass, reach_limit), NULL, NULL, g_cclosure_marshal_VOID__ENUM, G_TYPE_NONE, 1, GTK_TYPE_SCROLL_TYPE); g_signal_new("prepare-collapsing", GTK_TYPE_BUFFER_DISPLAY, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GtkBufferDisplayClass, prepare_collapsing), NULL, NULL, g_cclosure_marshal_VOID__BOOLEAN, G_TYPE_NONE, 1, G_TYPE_BOOLEAN); } /****************************************************************************** * * * Paramètres : display = composant GTK à initialiser. * * * * Description : Procède à l'initialisation de l'afficheur de tampons. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_init(GtkBufferDisplay *display) { display->cursor = NULL; } /****************************************************************************** * * * Paramètres : display = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_dispose(GtkBufferDisplay *display) { if (display->caret_timer != 0) { g_source_remove(display->caret_timer); display->caret_timer = 0; } g_clear_object(&display->view); g_clear_object(&display->cursor); g_clear_object(&display->builder); g_clear_object(&display->bar); G_OBJECT_CLASS(gtk_buffer_display_parent_class)->dispose(G_OBJECT(display)); } /****************************************************************************** * * * Paramètres : display = instance d'objet Gtk à traiter. * * * * Description : Procède à la libération totale de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_finalize(GtkBufferDisplay *display) { G_OBJECT_CLASS(gtk_buffer_display_parent_class)->finalize(G_OBJECT(display)); } /****************************************************************************** * * * Paramètres : widget = composant GTK visé par l'opération. * * event = informations liées à l'événement. * * * * Description : Intègre le focus dans le rendu du composant. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean gtk_buffer_display_focus(GtkWidget *widget, GdkEventFocus *event) { GtkBufferDisplay *display; /* Autre version du composant */ gboolean has_focus; /* Etat courant */ display = GTK_BUFFER_DISPLAY(widget); has_focus = event->in; if (has_focus) gtk_buffer_display_restart_caret_blinking(display); else if (display->caret_timer != 0) { g_source_remove(display->caret_timer); display->caret_timer = 0; display->show_caret = true; gtk_buffer_display_queue_draw_caret(display, NULL); } return TRUE; } /****************************************************************************** * * * Paramètres : widget = composant GTK visé par l'opération. * * event = informations liées à l'événement. * * * * Description : Assure la gestion des clics de souris sur le composant. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean gtk_buffer_display_button_press(GtkWidget *widget, GdkEventButton *event) { GtkBufferDisplay *display; /* Autre version du composant */ GBufferCache *cache; /* Contenu représenté */ gint left_margin; /* Limite entre zones réactives*/ gint real_x; /* Abscisse absolue réelle */ gint real_y; /* Ordonnée absolue réelle */ display = GTK_BUFFER_DISPLAY(widget); real_x = event->x; real_y = event->y; gtk_display_panel_compute_real_coord(GTK_DISPLAY_PANEL(display), &real_x, &real_y); cache = g_buffer_view_get_cache(display->view); left_margin = g_buffer_cache_get_left_margin(cache); g_object_unref(G_OBJECT(cache)); if (real_x < left_margin) { /* TODO */ } else _gtk_buffer_display_move_caret_to(display, real_x, real_y); gtk_widget_grab_focus(widget); return FALSE; } /****************************************************************************** * * * Paramètres : widget = composant GTK à redessiner. * * cr = contexte graphique associé à 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_display_draw(GtkWidget *widget, cairo_t *cr) { GtkBufferDisplay *display; /* Autre version du composant */ GtkDisplayPanel *parent; /* Autre version du composant */ GdkWindow *window; /* Fenêtre à redessiner */ GtkAllocation allocation; /* Aire complète du composant */ cairo_region_t *region; /* Région visible à redessiner */ cairo_rectangle_int_t area; /* Surface correspondante */ GtkStyleContext *context; /* Contexte du thème actuel */ gint virt_x; /* Abscisse virtuelle */ gint virt_y; /* Ordonnée virtuelle */ GBufferCache *cache; /* Contenu représenté */ gint left_margin; /* Marge gauche + espace */ bool sel_line; /* Souslignage de la sélection */ gint *selected; /* Ordonnée d'une sélection */ display = GTK_BUFFER_DISPLAY(widget); parent = GTK_DISPLAY_PANEL(widget); window = gtk_widget_get_window(widget); cairo_save(cr); gtk_cairo_transform_to_window(cr, widget, window); if (parent->export) { gtk_widget_get_allocation(widget, &allocation); area.x = 0; area.y = 0; area.width = allocation.width; area.height = allocation.height; } else { region = gdk_window_get_clip_region(window); cairo_region_get_extents(region, &area); cairo_region_destroy(region); } context = gtk_widget_get_style_context(widget); if (parent->show_border) { gtk_widget_get_allocation(widget, &allocation); allocation.x = 0; allocation.y = 0; gtk_display_panel_define_border_path(parent, cr, &allocation); cairo_clip(cr); } /* Décalage pour le défilement horizontal */ virt_x = 0; virt_y = 0; gtk_display_panel_compute_fake_coord(parent, &virt_x, &virt_y); cairo_save(cr); cairo_translate(cr, virt_x, 0); /* Récupération de la limite utile */ cache = g_buffer_view_get_cache(display->view); left_margin = g_buffer_cache_get_left_margin(cache) * parent->scale; g_object_unref(G_OBJECT(cache)); /* Dessin de la marge gauche */ gtk_style_context_save(context); gtk_style_context_add_class(context, GTK_STYLE_CLASS_SIDEBAR); gtk_render_background(context, cr, 0, area.y, left_margin, area.height); gtk_style_context_restore(context); /* Fond de la zone de texte */ gtk_style_context_save(context); gtk_style_context_add_class(context, GTK_STYLE_CLASS_VIEW); gtk_style_context_add_class(context, "graph-block-background"); gtk_render_background(context, cr, left_margin, area.y, area.width, area.height); gtk_style_context_restore(context); if (parent->show_border) { gtk_style_context_save(context); gtk_style_context_add_class(context, "graph-block"); gtk_render_background(context, cr, left_margin, area.y, area.width, area.height); gtk_style_context_restore(context); } /* Ligne de séparation */ gtk_style_context_save(context); gtk_style_context_add_class(context, GTK_STYLE_CLASS_FRAME); gtk_render_frame(context, cr, - 0.5, area.y - 1, left_margin + 0.5, area.height + 2); gtk_style_context_restore(context); /* Eventuelle bordure globale */ if (parent->show_border) gtk_display_panel_draw_border(parent, cr); /* Impression du désassemblage */ cairo_save(cr); cairo_scale(cr, parent->scale, parent->scale); if (display->view != NULL) { g_generic_config_get_value(get_main_configuration(), MPK_SELECTION_LINE, &sel_line); sel_line &= gtk_widget_has_focus(widget); if (!sel_line || display->cursor == NULL || !g_line_cursor_is_valid(display->cursor) || parent->export) selected = NULL; else { selected = (gint []) { display->caret.y }; gtk_display_panel_compute_relative_coords(parent, NULL, selected); } area.x -= virt_x; virt_y += area.y; g_buffer_view_draw(display->view, cr, virt_y, &area, parent->options, selected, parent->scale, parent->export); } cairo_restore(cr); /* Curseur clignotant ? */ cairo_restore(cr); if (gtk_widget_is_focus(widget)) gtk_buffer_display_draw_caret(display, cr); cairo_restore(cr); /* Dessin des composants contenus */ cairo_save(cr); cairo_scale(cr, parent->scale, parent->scale); GTK_WIDGET_CLASS(gtk_buffer_display_parent_class)->draw(widget, cr); cairo_restore(cr); return FALSE; } /****************************************************************************** * * * Paramètres : widget = composant visé par l'opération. * * event = informations liées à l'événement. * * * * Description : Prend en compte une frappe de touche sur le composant. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean gtk_buffer_display_key_press(GtkWidget *widget, GdkEventKey *event) { gboolean result; /* Suites à renvoyer */ GdkScrollDirection dir; /* Direction du déplacement */ GtkBufferDisplay *display; /* Autre version du composant */ GtkDisplayPanel *panel; /* Autre version du composant */ bool ctrl; /* Statut de la touche Contrôle*/ cairo_rectangle_int_t area; /* Emplacement de curseur #1 */ GLineCursor *cursor; /* Emplacement de curseur #2 */ bool status; /* Validité d'un déplacement */ switch (event->keyval) { case GDK_KEY_Left: dir = GDK_SCROLL_LEFT; result = TRUE; break; case GDK_KEY_Up: dir = GDK_SCROLL_UP; result = TRUE; break; case GDK_KEY_Right: dir = GDK_SCROLL_RIGHT; result = TRUE; break; case GDK_KEY_Down: dir = GDK_SCROLL_DOWN; result = TRUE; break; default: result = FALSE; break; } if (result) { display = GTK_BUFFER_DISPLAY(widget); panel = GTK_DISPLAY_PANEL(widget); ctrl = (event->state & GDK_CONTROL_MASK); area = display->caret; status = g_buffer_view_move_caret(display->view, ctrl, dir, panel->options, &area, &cursor); if (status) { gtk_buffer_display_relocate_caret(display, &area, cursor); g_loaded_panel_scroll_to_cursor(G_LOADED_PANEL(panel), cursor, SPT_RAW, false); } else g_signal_emit_by_name(display, "reach-limit", dir); } return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à consulter. * * width = largeur requise à renseigner ou NULL. [OUT] * * height = hauteur requise à renseigner ou NULL. [OUT] * * * * Description : Indique les dimensions de travail du composant d'affichage. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_compute_requested_size(GtkBufferDisplay *display, gint *width, gint *height) { gint extra; /* Eventuel supplément largeur */ if (width != NULL) { if (display->view != NULL) { *width = g_buffer_view_get_width(display->view, GTK_DISPLAY_PANEL(display)->options); if (display->bar != NULL) { gtk_widget_get_preferred_width(display->bar, NULL, &extra); *width += extra; } } else *width = 0; } if (height != NULL) { if (display->view != NULL) *height = g_buffer_view_get_height(display->view); else *height = 0; } } /****************************************************************************** * * * Paramètres : display = composant GTK d'affichage à consulter. * * size = taille de l'espace dans la direction donnée. * * orientation = indication sur le défilement à traiter. * * step = valeur d'un petit pas de défilement. [OUT] * * page = valeur d'un grand pas de défilement. [OUT] * * * * Description : Détermine la taille des bonds lors de défilements. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_compute_scroll_inc(GtkBufferDisplay *display, gint size, GtkOrientation orientation, gdouble *step, gdouble *page) { GBufferCache *cache; /* Gestionnaire de lignes */ if (orientation == GTK_ORIENTATION_VERTICAL && display->view != NULL) { cache = g_buffer_view_get_cache(display->view); *step = g_buffer_cache_get_line_height(cache); *page = *step * 10; g_object_unref(G_OBJECT(cache)); } else GTK_DISPLAY_PANEL_CLASS(gtk_buffer_display_parent_class)->compute_inc(GTK_DISPLAY_PANEL(display), size, orientation, step, page); } /****************************************************************************** * * * Paramètres : display = panneau d'affichage concerné. * * adj = défilement dont une valeur a changé. * * orientation = indication sur le défilement à traiter. * * * * Description : Réagit à un défilement chez une barre associée au composant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_adjust_scroll_value(GtkBufferDisplay *display, GtkAdjustment *adj, GtkOrientation orientation) { GtkWidget *widget; /* Autre vision du composant */ widget = GTK_WIDGET(display); if (gtk_widget_get_realized(widget)) gdk_window_invalidate_rect(gtk_widget_get_window(widget), NULL, FALSE); } /****************************************************************************** * * * Paramètres : display = composant GTK à consulter. * * cursor = emplacement à présenter à l'écran. * * x = position horizontale au sein du composant. [OUT] * * y = position verticale au sein du composant. [OUT] * * tweak = adaptation finale à effectuer. * * * * Description : Indique la position d'affichage d'un emplacement donné. * * * * Retour : true si l'adresse fait partie du composant, false sinon. * * * * Remarques : - * * * ******************************************************************************/ static bool gtk_buffer_display_get_cursor_coordinates(const GtkBufferDisplay *display, const GLineCursor *cursor, gint *x, gint *y, ScrollPositionTweak tweak) { bool result; /* Bilan à remonter */ bool need_code; /* Recherche plus raffinée */ GBufferCache *cache; /* Gestionnaire de lignes */ int height; /* Hauteur allouée */ need_code = (tweak == SPT_BOTTOM); cache = g_buffer_view_get_cache(display->view); result = g_buffer_view_get_cursor_coordinates(display->view, cursor, need_code, x, y); if (result) { *x += g_buffer_view_get_margin(display->view, GTK_DISPLAY_PANEL(display)->options); height = gtk_widget_get_allocated_height(GTK_WIDGET(display)); switch (tweak) { case SPT_RAW: break; case SPT_TOP: break; case SPT_CENTER: *y -= (height / 2); break; case SPT_BOTTOM: *y -= height; *y += g_buffer_cache_get_line_height(cache); break; } } g_object_unref(G_OBJECT(cache)); return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à consulter. * * * * Description : Fournit l'élément actif lié à la position courante. * * * * Retour : Objet actif courant ou NULL si aucun. * * * * Remarques : - * * * ******************************************************************************/ GObject *gtk_buffer_display_get_active_object(const GtkBufferDisplay *display) { GObject *result; /* Trouvaille à retourner */ /* Si aucune position n'est définie... */ if (display->cursor == NULL || !g_line_cursor_is_valid(display->cursor)) result = NULL; else result = g_buffer_view_find_creator(display->view, display->caret.x, display->caret.y, GTK_DISPLAY_PANEL(display)->options); return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * cairo = assistant pour la création de rendus. * * area = taille de la surface réduite à disposition. * * scale = échelle vis à vis de la taille réelle. * * * * Description : Place en cache un rendu destiné à l'aperçu graphique rapide. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_cache_glance(GtkBufferDisplay *display, cairo_t *cairo, const GtkAllocation *area, double scale) { cairo_set_line_width(cairo, 1); cairo_set_source_rgb(cairo, 0.4, 0.4, 0.4); cairo_rectangle(cairo, area->x + 0.5, area->y + 0.5, area->width - 1, area->height - 1); cairo_stroke(cairo); } /****************************************************************************** * * * Paramètres : display = composant GTK à consulter. * * * * Description : Fournit le position courante dans un panneau de chargement. * * * * Retour : Informations relatives à la position du curseur. * * * * Remarques : - * * * ******************************************************************************/ static GLineCursor *gtk_buffer_display_get_cursor(const GtkBufferDisplay *display) { GLineCursor *result; /* Contenu à retourner */ if (display->cursor != NULL) { result = display->cursor; g_object_ref(G_OBJECT(result)); } else result = g_binary_cursor_new(); return result; } /* ---------------------------------------------------------------------------------- */ /* ANIMATION DU CURSEUR */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : display = composant GTK à consulter. * * cursor = définition générique d'une localisation à l'écran. * * * * Description : Détermine si une position est comprise dans l'affichage. * * * * Retour : true si le composant comprend bien la localisation fournie. * * * * Remarques : - * * * ******************************************************************************/ bool gtk_buffer_display_contain_cursor(const GtkBufferDisplay *display, const GLineCursor *cursor) { bool result; /* Bilan à retourner */ GLineCursor *start; /* Position initiale du tampon */ GLineCursor *end; /* Position finale du tampon */ int status; /* Bilan d'une comparaison */ g_buffer_view_get_restrictions(display->view, &start, &end); if (start == NULL && end == NULL) result = NULL; else { status = g_line_cursor_compare(start, cursor); if (status > 0) result = false; else { status = g_line_cursor_compare(cursor, end); result = (status < 0); } g_object_unref(G_OBJECT(start)); g_object_unref(G_OBJECT(end)); } return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * x = abscisse proposée pour le nouvel emplacement. * * y = ordonnée proposée pour le nouvel emplacement. * * * * Description : Déplace le curseur à un emplacement défini. * * * * Retour : true si un traitement a été effectué, false sinon. * * * * Remarques : - * * * ******************************************************************************/ static bool _gtk_buffer_display_move_caret_to(GtkBufferDisplay *display, gint x, gint y) { bool result; /* Bilan à retourner */ GtkDisplayPanel *panel; /* Autre version du composant */ cairo_rectangle_int_t new; /* Nouvel emplacement calculé */ GLineCursor *cursor; /* Emplacement de curseur */ panel = GTK_DISPLAY_PANEL(display); result = g_buffer_view_compute_caret_full(display->view, x, y, panel->options, &new, &cursor); if (result) gtk_buffer_display_relocate_caret(display, &new, cursor); return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * beginning = précise le coin où se retrouvera le curseur. * * same_x = tente de conserver une même abscisse ou NULL ? * * * * Description : Déplace le curseur à un emplacement en extrémité. * * * * Retour : true si un traitement a été effectué, false sinon. * * * * Remarques : - * * * ******************************************************************************/ bool gtk_buffer_display_move_caret_to(GtkBufferDisplay *display, bool beginning, gint *same_x) { bool result; /* Bilan à remonter */ GBufferCache *cache; /* Contenu représenté */ gint left_margin; /* Limite entre zones réactives*/ gint x; /* Abscisse d'emplacement */ gint y; /* Ordonnée d'emplacement */ if (beginning) { cache = g_buffer_view_get_cache(display->view); left_margin = g_buffer_cache_get_left_margin(cache); g_object_unref(G_OBJECT(cache)); x = same_x != NULL ? *same_x : left_margin * 2; y = 0; } else { if (same_x != NULL) x = *same_x; else gtk_widget_get_preferred_width(GTK_WIDGET(display), NULL, &x); gtk_widget_get_preferred_height(GTK_WIDGET(display), NULL, &y); y--; } result = _gtk_buffer_display_move_caret_to(display, x, y); return result; } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * area = emplacement pour le dessin d'un curseur. * * cursor = emplacement représenté dans un tampon interne. * * * * Description : Déplace le curseur en effaçant son éventuelle position. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_relocate_caret(GtkBufferDisplay *display, const cairo_rectangle_int_t *area, GLineCursor *cursor) { bool clear_old; /* Effacement chirurgical */ cairo_rectangle_int_t old_area; /* Mémorisation de l'ancien */ bool need_redraw; /* Besoin de rafraîchissement ?*/ if (display->cursor != NULL && g_line_cursor_is_valid(display->cursor)) { clear_old = true; old_area = display->caret; } else clear_old = false; if (display->cursor != NULL) g_object_unref(G_OBJECT(display->cursor)); display->caret = *area; display->cursor = cursor; if (GTK_BUFFER_DISPLAY_GET_CLASS(display)->notify_caret != NULL) need_redraw = GTK_BUFFER_DISPLAY_GET_CLASS(display)->notify_caret(display, area); else need_redraw = false; if (display->cursor != NULL && g_line_cursor_is_valid(display->cursor)) gtk_buffer_display_restart_caret_blinking(display); if (need_redraw) gtk_buffer_display_queue_draw_caret(display, NULL); else if (clear_old) gtk_buffer_display_queue_draw_caret(display, &old_area); } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * * * Description : Assure le clignotement du curseur à l'emplacement courant. * * * * Retour : TRUE pour poursuivre le clignotement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean gtk_buffer_display_refresh_caret(GtkBufferDisplay *display) { /* Bascule l'affichage */ display->show_caret = !display->show_caret; assert(display->cursor != NULL && g_line_cursor_is_valid(display->cursor)); if (!display->show_caret) gtk_buffer_display_queue_draw_caret(display, &display->caret); else gtk_buffer_display_queue_draw_caret(display, NULL); return TRUE; } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * * * Description : Redémarre l'affichage du curseur à l'emplacement courant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_restart_caret_blinking(GtkBufferDisplay *display) { GtkSettings *settings; /* Propriétés du système */ guint interval; /* Fréquence d'actualisation */ if (display->caret_timer != 0) { g_source_remove(display->caret_timer); display->caret_timer = 0; } if (display->cursor != NULL && g_line_cursor_is_valid(display->cursor)) { settings = gtk_settings_get_default(); g_object_get(settings, "gtk-cursor-blink-time", &interval, NULL); display->show_caret = true; g_object_ref(G_OBJECT(display)); display->caret_timer = gdk_threads_add_timeout_full(G_PRIORITY_DEFAULT, interval, (GSourceFunc)gtk_buffer_display_refresh_caret, display, g_object_unref); } if (display->cursor != NULL) g_signal_emit_by_name(display, "cursor-moved", display->cursor); } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * area = emplacement du curseur ou NULL toute la région. * * * * Description : Prépare l'actualisation de tout ou une partie de l'affichage.* * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_queue_draw_caret(GtkBufferDisplay *display, cairo_rectangle_int_t *area) { GtkWidget *widget; /* Autre version du composant */ cairo_rectangle_int_t rect; /* Zone rectangulaire relative */ cairo_region_t *region; /* Zone précise à redessiner */ widget = GTK_WIDGET(display); if (area == NULL) gtk_widget_queue_draw(widget); else { rect = *area; gtk_display_panel_compute_relative_coords(GTK_DISPLAY_PANEL(display), &rect.x, &rect.y); region = cairo_region_create_rectangle(&rect); gtk_widget_queue_draw_region(widget, region); cairo_region_destroy(region); } /** * Pour une raison non comprise, le redessin n'est pris en compte que * si le parent est concerné également... */ widget = gtk_widget_get_parent(widget); if (GTK_IS_SCROLLED_WINDOW(widget)) gtk_widget_queue_draw(widget); } /****************************************************************************** * * * Paramètres : display = composant GTK à manipuler. * * cr = contexte graphique disponible pour l'opération. * * * * Description : Affiche le curseur à l'écran, s'il doit l'être. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_draw_caret(GtkBufferDisplay *display, cairo_t *cr) { cairo_rectangle_int_t area; /* Zone adaptée à traiter */ cairo_region_t *region; /* Région définie associée */ GtkWidget *widget; /* Autre version du composant */ GdkRGBA *color; /* Couleur du curseur */ if (display->cursor != NULL && g_line_cursor_is_valid(display->cursor) && display->show_caret) { area = display->caret; gtk_display_panel_compute_relative_coords(GTK_DISPLAY_PANEL(display), &area.x, &area.y); region = cairo_region_create_rectangle(&area); widget = GTK_WIDGET(display); gtk_style_context_get(gtk_widget_get_style_context(widget), gtk_widget_get_state_flags(widget), GTK_STYLE_PROPERTY_COLOR, &color, NULL); cairo_set_source_rgb(cr, color->red, color->green, color->blue); cairo_rectangle(cr, area.x, area.y, area.width, area.height); cairo_fill(cr); cairo_region_destroy(region); } } /* ---------------------------------------------------------------------------------- */ /* INCLUSION D'UNE BARRE D'OUTILS */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : display = panneau d'affichage concerné. * * * * Description : Ajoute une nouvelle barre d'outils pour bloc au composant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_buffer_display_add_block_bar(GtkBufferDisplay *display) { GtkWidget *bar; /* Barre d'outils à intégrer */ assert(display->builder == NULL); display->builder = gtk_builder_new_from_resource("/org/chrysalide/gtkext/blockbar.ui"); bar = GTK_WIDGET(gtk_builder_get_object(display->builder, "blockbar")); display->bar = bar; g_object_ref(G_OBJECT(bar)); g_object_ref(G_OBJECT(bar)); gtk_widget_unparent(bar); gtk_fixed_put(GTK_FIXED(display), bar, 0, 0); gtk_builder_add_callback_symbols(display->builder, BUILDER_CALLBACK(on_block_bar_enter_notify), BUILDER_CALLBACK(on_block_bar_leave_notify), BUILDER_CALLBACK(on_block_bar_collapsing_enter), BUILDER_CALLBACK(on_block_bar_collapsing_leave), NULL); gtk_builder_connect_signals(display->builder, display); gtk_buffer_display_move_block_bar(display); } /****************************************************************************** * * * Paramètres : display = panneau d'affichage concerné. * * * * Description : Place correctement la barre d'outils pour bloc. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_buffer_display_move_block_bar(GtkBufferDisplay *display) { GtkWidget *bar; /* Barre d'outils courante */ gint width; /* Largeur requise à vide */ bar = display->bar; display->bar = NULL; gtk_buffer_display_compute_requested_size(display, &width, NULL); display->bar = bar; gtk_fixed_move(GTK_FIXED(display), bar, width, (int)BORDER_CORNER_RADIUS / 2); } /****************************************************************************** * * * Paramètres : widget = composant graphique concerné par l'opération. * * event = informations liées à l'événement. * * display = panneau d'affichage impliqué par l'action. * * * * Description : Accompagne le début du survol d'un élément de barre d'outils.* * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_block_bar_enter_notify(GtkWidget *widget, GdkEventCrossing *event, GtkBufferDisplay *display) { gtk_widget_set_opacity(widget, 1.0); return FALSE; } /****************************************************************************** * * * Paramètres : widget = composant graphique concerné par l'opération. * * event = informations liées à l'événement. * * display = panneau d'affichage impliqué par l'action. * * * * Description : Accompagne la fin du survol d'un élément de barre d'outils. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_block_bar_leave_notify(GtkWidget *widget, GdkEventCrossing *event, GtkBufferDisplay *display) { gtk_widget_set_opacity(widget, 0.6); return FALSE; } /****************************************************************************** * * * Paramètres : widget = composant graphique concerné par l'opération. * * event = informations liées à l'événement. * * display = panneau d'affichage impliqué par l'action. * * * * Description : Accompagne le début du survol du bouton de compression. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_block_bar_collapsing_enter(GtkWidget *widget, GdkEventCrossing *event, GtkBufferDisplay *display) { g_signal_emit_by_name(display, "prepare-collapsing", FALSE); return FALSE; } /****************************************************************************** * * * Paramètres : widget = composant graphique concerné par l'opération. * * event = informations liées à l'événement. * * display = panneau d'affichage impliqué par l'action. * * * * Description : Accompagne la fin du survol du bouton de compression. * * * * Retour : FALSE pour poursuivre la propagation de l'événement. * * * * Remarques : - * * * ******************************************************************************/ static gboolean on_block_bar_collapsing_leave(GtkWidget *widget, GdkEventCrossing *event, GtkBufferDisplay *display) { g_signal_emit_by_name(display, "prepare-collapsing", TRUE); return FALSE; } #endif