/* Chrysalide - Outil d'analyse de fichiers binaires * bufferline.c - représentation de fragments de texte en ligne * * Copyright (C) 2010-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 Chrysalide. If not, see <http://www.gnu.org/licenses/>. */ #include "bufferline.h" #include <assert.h> #include <malloc.h> #include <string.h> #include "chrysamarshal.h" #include "linecolumn.h" #include "../common/extstr.h" #include "../core/paths.h" #include "../gtkext/gtkblockdisplay.h" /* ---------------------------- GESTION DE LINE COMPLETE ---------------------------- */ /* Mémorisation des origines de texte */ typedef struct _content_origin { col_coord_t coord; /* Localisation d'attachement */ GObject *creator; /* Origine de la création */ } content_origin; /* Représentation de fragments de texte en ligne (instance) */ struct _GBufferLine { GObject parent; /* A laisser en premier */ line_column *columns; /* Répartition du texte */ size_t col_count; /* Nombre de colonnes présentes*/ size_t merge_start; /* Début de la zone globale */ BufferLineFlags flags; /* Drapeaux particuliers */ content_origin *origins; /* Mémorisation des origines */ size_t ocount; /* Nombre de ces mémorisations */ }; /* Représentation de fragments de texte en ligne (classe) */ struct _GBufferLineClass { GObjectClass parent; /* A laisser en premier */ cairo_surface_t *entrypoint_img; /* Image pour les entrées */ cairo_surface_t *bookmark_img; /* Image pour les signets */ /* Signaux */ void (* content_changed) (GBufferLine *, line_segment *); void (* flip_flag) (GBufferLine *, BufferLineFlags, BufferLineFlags); }; /* Procède à l'initialisation d'une classe de représentation. */ static void g_buffer_line_class_init(GBufferLineClass *); /* Procède à l'initialisation d'une représentation de fragments. */ static void g_buffer_line_init(GBufferLine *); /* Supprime toutes les références externes. */ static void g_buffer_line_dispose(GBufferLine *); /* Procède à la libération totale de la mémoire. */ static void g_buffer_line_finalize(GBufferLine *); /* ---------------------------------------------------------------------------------- */ /* GESTION DE LINE COMPLETE */ /* ---------------------------------------------------------------------------------- */ /* Détermine le type de la représentation de fragments de texte en ligne. */ G_DEFINE_TYPE(GBufferLine, g_buffer_line, G_TYPE_OBJECT); /****************************************************************************** * * * Paramètres : class = classe de composant GTK à initialiser. * * * * Description : Procède à l'initialisation d'une classe de représentation. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_buffer_line_class_init(GBufferLineClass *class) { GObjectClass *object; /* Autre version de la classe */ gchar *filename; /* Chemin d'accès à utiliser */ object = G_OBJECT_CLASS(class); object->dispose = (GObjectFinalizeFunc/* ! */)g_buffer_line_dispose; object->finalize = (GObjectFinalizeFunc)g_buffer_line_finalize; filename = find_pixmap_file("entrypoint.png"); assert(filename != NULL); class->entrypoint_img = cairo_image_surface_create_from_png(filename); g_free(filename); filename = find_pixmap_file("bookmark.png"); assert(filename != NULL); class->bookmark_img = cairo_image_surface_create_from_png(filename); g_free(filename); g_signal_new("content-changed", G_TYPE_BUFFER_LINE, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GBufferLineClass, content_changed), NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, G_TYPE_OBJECT); g_signal_new("flip-flag", G_TYPE_BUFFER_LINE, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GBufferLineClass, flip_flag), NULL, NULL, g_cclosure_user_marshal_VOID__ENUM_ENUM, G_TYPE_NONE, 2, G_TYPE_UINT, G_TYPE_UINT); } /****************************************************************************** * * * Paramètres : line = composant GTK à initialiser. * * * * Description : Procède à l'initialisation d'une représentation de fragments.* * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_buffer_line_init(GBufferLine *line) { line->columns = NULL; line->col_count = 0; line->merge_start = -1; } /****************************************************************************** * * * Paramètres : line = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_buffer_line_dispose(GBufferLine *line) { size_t i; /* Boucle de parcours */ for (i = 0; i < line->ocount; i++) g_object_unref(G_OBJECT(line->origins[i].creator)); G_OBJECT_CLASS(g_buffer_line_parent_class)->dispose(G_OBJECT(line)); } /****************************************************************************** * * * Paramètres : line = instance d'objet GLib à traiter. * * * * Description : Procède à la libération totale de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void g_buffer_line_finalize(GBufferLine *line) { size_t i; /* Boucle de parcours */ for (i = 0; i < line->col_count; i++) reset_line_column(&line->columns[i]); if (line->columns != NULL) free(line->columns); if (line->origins != NULL) free(line->origins); G_OBJECT_CLASS(g_buffer_line_parent_class)->finalize(G_OBJECT(line)); } /****************************************************************************** * * * Paramètres : col_count = quantité de colonnes à considérer. * * * * Description : Crée une nouvelle représentation de fragments de texte. * * * * Retour : Composant GTK créé. * * * * Remarques : - * * * ******************************************************************************/ GBufferLine *g_buffer_line_new(size_t col_count) { GBufferLine *result; /* Composant à retourner */ size_t i; /* Boucle de parcours */ result = g_object_new(G_TYPE_BUFFER_LINE, NULL); result->columns = malloc(col_count * sizeof(line_column)); for (i = 0; i < col_count; i++) init_line_column(&result->columns[i]); result->col_count = col_count; return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * col = indice de la colonne à constituer. * * size = taille souhaitée de l'impression des positions. * * addr = localisation physique à venir représenter. * * * * Description : Construit le tronc commun d'une ligne autour de sa position. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_fill_phys(GBufferLine *line, size_t col, MemoryDataSize size, const vmpa2t *addr) { VMPA_BUFFER(position); /* Emplacement au format texte */ size_t len; /* Taille de l'élément inséré */ size_t i; /* Boucle de parcours #1 */ vmpa2_phys_to_string(addr, size, position, &len); for (i = 2; i < len; i++) if (position[i] != '0') break; if (i == len) i = len - 1; if (i > 0) g_buffer_line_append_text(line, col, position, i, RTT_PHYS_ADDR_PAD, NULL); g_buffer_line_append_text(line, col, &position[i], len - i, RTT_PHYS_ADDR, NULL); } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * col = indice de la colonne à constituer. * * size = taille souhaitée de l'impression des positions. * * addr = localisation virtuelle à venir représenter. * * * * Description : Construit le tronc commun d'une ligne autour de sa position. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_fill_virt(GBufferLine *line, size_t col, MemoryDataSize size, const vmpa2t *addr) { VMPA_BUFFER(position); /* Emplacement au format texte */ size_t len; /* Taille de l'élément inséré */ size_t i; /* Boucle de parcours #1 */ vmpa2_virt_to_string(addr, size, position, &len); if (has_virt_addr(addr)) { for (i = 2; i < len; i++) if (position[i] != '0') break; if (i == len) i = len - 1; if (i > 0) g_buffer_line_append_text(line, col, position, i, RTT_VIRT_ADDR_PAD, NULL); g_buffer_line_append_text(line, col, &position[i], len - i, RTT_VIRT_ADDR, NULL); } else g_buffer_line_append_text(line, col, position, len, RTT_VIRT_ADDR_PAD, NULL); } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * col = indice de la colonne à constituer. * * content = contenu binaire global à venir lire. * * range = localisation des données à venir lire et présenter.* * max = taille maximale de la portion binaire en octets. * * * * Description : Construit le tronc commun d'une ligne autour de son contenu. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_fill_content(GBufferLine *line, size_t col, const GBinContent *content, const mrange_t *range, phys_t max) { phys_t length; /* Taille de la couverture */ bool truncated; /* Indique si le code est coupé*/ size_t required; /* Taille de traitement requise*/ char static_buffer[64]; /* Petit tampon local rapide */ char *bin_code; /* Tampon utilisé pour le code */ vmpa2t pos; /* Boucle de parcours #1 */ phys_t i; /* Boucle de parcours #2 */ char *iter; /* Boucle de parcours #3 */ int ret; /* Progression dans l'écriture */ uint8_t byte; /* Octet à représenter */ static const char *charset = "0123456789abcdef"; /* Détermination du réceptacle */ length = get_mrange_length(range); truncated = (max != VMPA_NO_PHYSICAL && length > max); if (truncated) { length = max; required = length * 3 + 4 /* "..." */ + 1; } else required = length * 3 + 1; if (required <= sizeof(static_buffer)) bin_code = static_buffer; else bin_code = (char *)calloc(required, sizeof(char)); /* Code brut */ copy_vmpa(&pos, get_mrange_addr(range)); for (i = 0, iter = bin_code; i < length; i++, iter += ret) { if (i == 0) ret = 0; else { iter[0] = ' '; ret = 1; } if (!g_binary_content_read_u8(content, &pos, &byte)) { iter[ret + 0] = '?'; iter[ret + 1] = '?'; } else { iter[ret + 0] = charset[byte >> 4]; iter[ret + 1] = charset[byte & 0x0f]; } ret += 2; } if (truncated) { strcpy(iter, "..."); iter += 3; } else *iter = '\0'; /* Conclusion */ g_buffer_line_append_text(line, col, bin_code, iter - bin_code, RTT_RAW_CODE, NULL); if (bin_code != static_buffer) free(bin_code); } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * column = indice de la colonne visée par les recherches. * * * * Description : Recherche le premier créateur enregistré dans des segments. * * * * Retour : Créateur trouvé à déréférencer par la suite ou NULL si échec.* * * * Remarques : - * * * ******************************************************************************/ GObject *g_buffer_line_find_first_segment_creator(const GBufferLine *line, size_t column) { GObject *result; /* Trouvaille à retourner */ size_t i; /* Boucle de parcours */ assert(column < line->col_count); result = NULL; for (i = 0; i < line->ocount && result == NULL; i++) { if (line->origins[i].coord.column == column) result = line->origins[i].creator; } if (result != NULL) g_object_ref(G_OBJECT(result)); return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * column = colonne de la ligne visée par l'insertion. * * text = texte à insérer dans l'existant. * * length = taille du texte à traiter. * * type = type de décorateur à utiliser. * * creator = instance GLib quelconque à associer. * * * * Description : Ajoute du texte à formater dans une ligne donnée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_append_text(GBufferLine *line, size_t column, const char *text, size_t length, RenderingTagType type, GObject *creator) { size_t index; /* Indice d'insertion */ content_origin *origin; /* Définition d'une origine */ assert(column < line->col_count); assert(length > 0); index = append_text_to_line_column(&line->columns[column], text, length, type); if (creator != NULL) { line->origins = realloc(line->origins, ++line->ocount * sizeof(content_origin)); origin = &line->origins[line->ocount - 1]; origin->coord.column = column; origin->coord.index = index; origin->creator = creator; g_object_ref(G_OBJECT(creator)); } } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * creator = instance GLib quelconque identifiant un segment. * * text = texte à insérer dans l'existant. * * length = taille du texte à traiter. * * * * Description : Remplace du texte dans une ligne donnée. * * * * Retour : Bilan de l'opération. * * * * Remarques : - * * * ******************************************************************************/ bool g_buffer_line_replace_text(GBufferLine *line, const GObject *creator, const char *text, size_t length) { bool result; /* Bilan à retourner */ size_t i; /* Boucle de parcours */ const col_coord_t *coord; /* Emplacement du contenu visé */ result = false; for (i = 0; i < line->ocount && !result; i++) { if (line->origins[i].creator == creator) { coord = &line->origins[i].coord; replace_text_in_line_column(&line->columns[coord->column], coord->index, text, length); g_signal_emit_by_name(line, "content-changed", NULL); result = true; } } return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * first = première colonne à parcourir. * * end = colonne de fin de parcours. * * * * Description : Indique si du texte est présent dans une ligne de tampon. * * * * Retour : true pour indiquer la présence de texte, false pour du vide. * * * * Remarques : - * * * ******************************************************************************/ bool g_buffer_line_has_text(const GBufferLine *line, size_t first, size_t end) { bool result; /* Bilan à retourner */ size_t i; /* Boucle de parcours */ result = false; assert(first < end); for (i = first; i < end && !result; i++) result = (line->columns[i].count > 0); return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * first = première colonne à parcourir. * * end = colonne de fin de parcours. * * markup = indique si le texte doit être décoré ou non. * * * * Description : Donne le texte représenté par une ligne de tampon. * * * * Retour : Texte à libérer de la mémoire après usage. * * * * Remarques : - * * * ******************************************************************************/ char *g_buffer_line_get_text(const GBufferLine *line, size_t first, size_t end, bool markup) { char *result; /* Construction à retourner */ size_t i; /* Boucle de parcours */ char *extra; /* Contenu à intégrer au texte */ result = NULL; assert(first < end); for (i = first; i < end; i++) { if (i > first && result != NULL) result = stradd(result, " "); extra = get_line_column_text(&line->columns[i], markup); /* Si la colonne était vide, suivante ! */ if (extra == NULL) continue; if (result == NULL) result = extra; else { result = stradd(result, extra); free(extra); } } return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir modifier. * * first = première colonne à parcourir. * * end = colonne de fin de parcours. * * * * Description : Supprime du texte représenté par une ligne de tampon. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_delete_text(GBufferLine *line, size_t first, size_t end) { size_t i; /* Boucle de parcours */ assert(first < end); for (i = first; i < end; i++) reset_line_column(&line->columns[i]); } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * * * Description : Fournit la colonne à partir de laquelle une fusion opère. * * * * Retour : Début de la première (et unique) zone globale. * * * * Remarques : - * * * ******************************************************************************/ size_t g_buffer_line_get_merge_start(const GBufferLine *line) { return line->merge_start; } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * start = début de la première (et unique) zone globale. * * * * Description : Définit la colonne à partir de laquelle la fusion opère. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_start_merge_at(GBufferLine *line, size_t start) { line->merge_start = start; } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * flag = propriété à intégrer. * * * * Description : Ajoute une propriété particulière à une ligne donnée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_add_flag(GBufferLine *line, BufferLineFlags flag) { if ((line->flags & flag) == 0) { g_signal_emit_by_name(line, "flip-flag", line->flags, flag); line->flags |= flag; } } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * * * Description : Renseigne sur les propriétés particulières liées à une ligne.* * * * Retour : Propriétés intégrées. * * * * Remarques : - * * * ******************************************************************************/ BufferLineFlags g_buffer_line_get_flags(const GBufferLine *line) { return line->flags; } /****************************************************************************** * * * Paramètres : line = ligne à venir compléter. * * flag = propriété à supprimer. * * * * Description : Retire une propriété particulière à une ligne donnée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_remove_flag(GBufferLine *line, BufferLineFlags flag) { if ((line->flags & flag) != 0) { g_signal_emit_by_name(line, "flip-flag", line->flags, flag); line->flags &= ~flag; } } /****************************************************************************** * * * Paramètres : line = ligne de texte à manipuler. * * ctx = éléments à disposition pour l'exportation. * * type = type d'exportation attendue. * * col_count = quantité de colonnes existantes au total. * * options = règles d'affichage des colonnes modulables. * * * * Description : Exporte la ligne de texte représentée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_export(GBufferLine *line, buffer_export_context *ctx, BufferExportType type, const GDisplayOptions *options) { size_t opt_count; /* Qté de colonnes en option */ size_t i; /* Boucle de parcours */ int col_span; /* Fusion de colonnes ? */ switch (type) { case BET_HTML: dprintf(ctx->fd, "\t<TR>\n"); break; default: break; } opt_count = g_display_options_count(options); assert(opt_count < line->col_count); for (i = 0; i < line->col_count; i++) { if (i < opt_count) { if (!g_display_options_get(options, i)) continue; } switch (type) { case BET_TEXT: if (i > 0) dprintf(ctx->fd, "%s", ctx->sep); break; default: break; } /** * Pour la signification des différentes valeurs assignées, * se référer au code de export_line_column_segments(). * * En gros : * - 1 = rien de spécial. * - >1 = il s'agit de la première cellule fusionnée de la ligne. * - 0 = fusion déjà faite, on ne peut que rajouter du contenu dedans. * - <1 = il s'agit de la dernière cellule fusionnée de la ligne. * * On considère qu'une fusion ne peut pas se réaliser sur la dernière * cellule uniquement (ce qui a du sens : c'est inutile). */ if (i < line->merge_start) col_span = 1; else if (i == line->merge_start) col_span = line->col_count - i; else col_span = ((i + 1) == line->col_count ? -1 : 0); export_line_column_segments(&line->columns[i], ctx, type, col_span); } switch (type) { case BET_TEXT: dprintf(ctx->fd, "\n"); break; case BET_HTML: dprintf(ctx->fd, "</TR>\n"); break; default: break; } } /*----------------------------------------------------------------------------------- */ /* MANIPULATION DES LARGEURS REQUISES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * col_count = quantité de colonnes à considérer. * * opt_count = quantité de colonnes optionnelles. * * widths = largeur mesurée pour chacune des colonnes. [OUT] * * merged = largeur cumulée en cas de fusion. [OUT] * * * * Description : Fait remonter les largeurs requises par une ligne donnée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_collect_widths(const GBufferLine *line, size_t col_count, size_t opt_count, gint *widths, gint *merged) { size_t i; /* Boucle de parcours */ gint width; /* Largeur d'une colonne */ assert(col_count == line->col_count); *merged = 0; for (i = 0; i < col_count; i++) { width = get_column_width(&line->columns[i]); widths[i] = (i < line->merge_start ? width : 0); if (line->merge_start != -1 && i >= opt_count) { *merged += width; if (i < line->merge_start && (i + 1) < col_count) *merged += COL_MARGIN; } } } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * coord = coordonnées interne du segment à retrouver. * * * * Description : Fournit le segment présent à une position donnée. * * * * Retour : Segment trouvé ou NULL si hors borne. * * * * Remarques : - * * * ******************************************************************************/ line_segment *g_buffer_line_get_segment_from_coord(const GBufferLine *line, const col_coord_t *coord) { line_segment *result; /* Trouvaille à retourner */ if (coord->column < line->col_count) result = get_line_column_content_from_index(&line->columns[coord->column], coord->index); else result = NULL; return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * index = indice de ligne associé. * * tracker = gestionnaire de largeur à consulter au besoin. * * options = règles d'affichage des colonnes modulables. * * offsets = décalages supplémentaires à appliquer. * * base = position jusqu'au segment trouvé. [OUT] * * offset = position à la colonne visée. [OUT] * * dir = direction d'un éventuel déplacement en cours. * * force = accepte les segments en bordure au pire. * * coord = cordonnées à usage interne à renseigner. [OUT] * * * * Description : Fournit les coordonnées correspondant à une abscisse donnée. * * * * Retour : true si des coordonnées valides ont été renseignées. * * * * Remarques : - * * * ******************************************************************************/ bool g_buffer_line_get_coord_at(const GBufferLine *line, size_t index, GWidthTracker *tracker, const GDisplayOptions *options, gint *base, gint *offset, GdkScrollDirection dir, bool force, col_coord_t *coord) { bool result; /* Bilan à retourner */ size_t last; /* Dernière colonne remplie */ gint last_base; /* Dernière abscisse associée */ size_t col_count; /* Qté de colonnes présentes */ size_t opt_count; /* Qté de colonnes en option */ size_t i; /* Boucle de parcours */ gint width; /* Largeur d'une colonne donnée*/ gint limit; /* Limite d'appartenance */ gint consumed; /* Distance vers le segment */ gint old_base; /* Somme de toutes les largeurs*/ result = false; *base = 0; last = line->col_count; last_base = 0; /* On cible déjà la colonne idéale */ col_count = g_width_tracker_count_columns(tracker); opt_count = g_display_options_count(options); for (i = 0; i < col_count; i++) { if (i < opt_count) { if (!g_display_options_get(options, i)) continue; } /* Mémorisation de la dernière colonne contenant quelque chose... */ if (get_column_width(&line->columns[i]) > 0) { last = i; last_base = *base; } if (i < line->merge_start) { width = g_width_tracker_get_local_column_width(tracker, index, i, opt_count); /* Si la colonne n'est absolument pas visible, on ne s'arrête pas dessus ! */ if (width == 0) continue; if ((i + 1) < col_count) limit = width + COL_MARGIN / 2; else limit = width; if (*offset <= limit) break; else { *offset -= width + COL_MARGIN; *base += width + COL_MARGIN; } } else { width = get_column_width(&line->columns[i]); if (*offset <= width) break; else { *offset -= width; *base += width; } } } /* Si l'abscisse fournie tombe encore dans une colonne... */ if (i < col_count) { /* Il y a bien du contenu dans cette colonne */ if (get_column_width(&line->columns[i]) > 0) { /** * Si la position était au milieu d'une marge, la sélection a pu pousser * jusqu'à la colonne suivante, plus proche. * Relativment à la base de cette dernière, la position est donc devenue négative. */ if (*offset < 0) *offset = 0; result = get_line_column_content_index_at(&line->columns[i], offset, dir, &consumed, &coord->index); if (result) { coord->column = i; *base += consumed; } } /* La position fournie tombe dans une colonne vide ! */ else { if (force || get_column_width(&line->columns[i]) == 0) { result = false; *offset = 0; old_base = *base; for (i++; i < col_count && !result; i++) { if ((i - 1) < line->merge_start) { width = g_width_tracker_get_local_column_width(tracker, index, i - 1, opt_count); if (width > 0) *base += (width + COL_MARGIN); } else *base += get_column_width(&line->columns[i - 1]); result = get_line_column_first_content_index(&line->columns[i], &coord->index); if (result) coord->column = i; } if (!result) { *base = old_base; goto use_right_border; } } } } else /* if (i == col_count) */ { if (force) { use_right_border: if (last != col_count) { result = get_line_column_last_content_index(&line->columns[last], &coord->index); if (result) { coord->column = last; *base = last_base; *offset = get_column_width(&line->columns[last]); } } /* Il n'y a rien sur la ligne ! */ else { result = true; *base = 0; *offset = 0; coord->column = col_count; coord->index = -1; } } else result = false; } return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * index = indice de ligne associé. * * tracker = gestionnaire de largeur à consulter au besoin. * * options = règles d'affichage des colonnes modulables. * * base = position jusqu'au segment trouvé. [OUT] * * offset = position à la colonne visée. [OUT] * * dir = direction d'un éventuel déplacement en cours. * * force = accepte les segments en bordure au pire. * * * * Description : Donne le segment présent à une abscisse donnée. * * * * Retour : Segment trouvé ou NULL si hors borne. * * * * Remarques : - * * * ******************************************************************************/ line_segment *g_buffer_line_get_segment_at(const GBufferLine *line, size_t index, GWidthTracker *tracker, const GDisplayOptions *options, gint *base, gint *offset, GdkScrollDirection dir, bool force) { line_segment *result; /* Trouvaille à retourner */ col_coord_t coord; /* Emplacement du contenu visé */ bool status; /* Bilan de la localisation */ status = g_buffer_line_get_coord_at(line, index, tracker, options, base, offset, dir, force, &coord); if (status) result = g_buffer_line_get_segment_from_coord(line, &coord); else result = NULL; return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * index = indice de ligne associé. * * tracker = gestionnaire de largeur à consulter au besoin. * * options = règles d'affichage des colonnes modulables. * * base = position jusqu'au segment trouvé. [OUT] * * offset = position à la colonne visée. [OUT] * * dir = direction d'un éventuel déplacement en cours. * * force = accepte les segments en bordure au pire. * * * * Description : Donne le créateur présent à une abscisse donnée. * * * * Retour : Créateur trouvé ou NULL si hors borne. * * * * Remarques : - * * * ******************************************************************************/ GObject *g_buffer_line_get_creator_at(const GBufferLine *line, size_t index, GWidthTracker *tracker, const GDisplayOptions *options, gint *base, gint *offset, GdkScrollDirection dir, bool force) { GObject *result; /* Trouvaille à retourner */ col_coord_t target; /* Emplacement du contenu visé */ bool status; /* Bilan de la localisation */ size_t i; /* Boucle de parcours */ const col_coord_t *coord; /* Emplacement du contenu visé */ result = NULL; status = g_buffer_line_get_coord_at(line, index, tracker, options, base, offset, dir, force, &target); if (status) { for (i = 0; i < line->ocount && result == NULL; i++) { coord = &line->origins[i].coord; if (coord->column == target.column && coord->index == target.index) result = line->origins[i].creator; } if (result != NULL) g_object_ref(G_OBJECT(result)); } return result; } /****************************************************************************** * * * Paramètres : line = ligne à venir consulter. * * index = indice de ligne associé. * * coord = cordonnées à consulter puis renseigner. [OUT] * * tracker = gestionnaire de largeur à consulter au besoin. * * options = règles d'affichage des colonnes modulables. * * dir = orientation des recherches. * * offset = décalage pour amener à l'extrémité nouvelle. [OUT] * * * * Description : Fournit des coordonnées voisines selon une direction donnée. * * * * Retour : true si des coordonnées valides ont été renseignées. * * * * Remarques : - * * * ******************************************************************************/ bool g_buffer_line_find_near_coord(const GBufferLine *line, size_t index, col_coord_t *coord, GWidthTracker *tracker, const GDisplayOptions *options, GdkScrollDirection dir, gint *offset) { bool result; /* Bilan à retourner */ size_t col_count; /* Qté de colonnes présentes */ size_t i; /* Boucle de parcours #1 */ size_t opt_count; /* Qté de colonnes en option */ bool displayed; /* Confort de lecture */ size_t k; /* Boucle de parcours #2 */ gint width; /* Largeur d'une colonne donnée*/ result = false; col_count = g_width_tracker_count_columns(tracker); /* Recherche dans la colonne de départ */ i = coord->column; if (i == col_count) return false; result = find_near_segment(&line->columns[i], &coord->index, dir); /* Recherche dans la direction des colonnes voisines */ opt_count = g_display_options_count(options); if (!result) switch (dir) { case GDK_SCROLL_LEFT: /* Si on a atteint la première colonne sans trouver... */ if (i == 0) break; /* On s'assure que la colonne précédente est visible et peuplée */ for (; i > 0 && !result; i--) { displayed = (i <= opt_count ? g_display_options_get(options, i - 1) : true); if (displayed) { result = get_line_column_first_content_index(&line->columns[i - 1], &coord->index); if (result) coord->column = i - 1; } } break; case GDK_SCROLL_RIGHT: /* On s'assure que la colonne suivante est visible et peuplée */ for (; (i + 1) < col_count && !result; i++) { displayed = ((i + 1) < opt_count ? g_display_options_get(options, i + 1) : true); if (displayed) { result = get_line_column_first_content_index(&line->columns[i + 1], &coord->index); if (result) coord->column = i + 1; } } break; default: break; } /* Calcul de la position finale */ if (result) { *offset = 0; for (k = 0; k < i; k++) { displayed = (k < opt_count ? g_display_options_get(options, k) : true); if (displayed) { if (k >= line->merge_start) width = get_column_width(&line->columns[index]); else width = g_width_tracker_get_local_column_width(tracker, index, k, opt_count); if (width > 0) { *offset += width; if (k < line->merge_start) *offset += COL_MARGIN; } } } switch (dir) { case GDK_SCROLL_LEFT: *offset += get_column_width(&line->columns[i]); break; case GDK_SCROLL_RIGHT: /**offset += 0;*/ break; default: break; } } return result; } /****************************************************************************** * * * Paramètres : line = ligne de texte à manipuler. * * index = indice de ligne associé. * * cairo = contexte graphique à utiliser pour les pinceaux. * * x_init = abscisse du point d'impression de départ. * * y = ordonnée du point d'impression. * * tracker = gestionnaire de largeur à consulter au besoin. * * options = règles d'affichage des colonnes modulables. * * list = liste de contenus à mettre en évidence. * * * * Description : Imprime la ligne de texte représentée. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void g_buffer_line_draw(GBufferLine *line, size_t index, cairo_t *cairo, gint x_init, gint y, GWidthTracker *tracker, const GDisplayOptions *options, const segcnt_list *list) { GBufferLineClass *class; /* Stockage de briques de base */ bool has_src_surface; /* Note une présence définie */ gint x; /* Point de départ d'impression*/ size_t col_count; /* Qté de colonnes présentes */ size_t opt_count; /* Qté de colonnes en option */ size_t i; /* Boucle de parcours */ gint max_width; /* Largeur maximale de colonne */ if (line->flags != BLF_NONE) { class = G_BUFFER_LINE_GET_CLASS(line); if (line->flags & BLF_ENTRYPOINT) { cairo_set_source_surface(cairo, class->entrypoint_img, 5, y); has_src_surface = true; } else if (line->flags & BLF_BOOKMARK) { cairo_set_source_surface(cairo, class->bookmark_img, 5, y); has_src_surface = true; } else has_src_surface = false; if (has_src_surface) cairo_paint(cairo); } x = x_init; col_count = g_width_tracker_count_columns(tracker); opt_count = g_display_options_count(options); for (i = 0; i < col_count; i++) { if (i < opt_count) { if (!g_display_options_get(options, i)) continue; } draw_line_column_segments(&line->columns[i], cairo, x, y, list); if (i < line->merge_start) { max_width = g_width_tracker_get_local_column_width(tracker, index, i, opt_count); if (max_width > 0) x += max_width + COL_MARGIN; } } }