/* 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 .
*/
#include "bufferline.h"
#include
#include
#include
#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\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, "
\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 && line->flags != BLF_HAS_CODE)
{
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;
}
}
}