/* Chrysalide - Outil d'analyse de fichiers binaires * leaving.c - liens sortants d'un bloc de code dans une représentation graphique * * Copyright (C) 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 "leaving.h" #include #include #include "cluster-int.h" #include "incoming.h" /****************************************************************************** * * * Paramètres : owner = propriétaire du bloc de rattachement. * * index = indice dans les liens de sortie. * * * * Description : Crée un point d'attache pour un lien sortant. * * * * Retour : Structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ leaving_link_t *create_leaving_link(GGraphCluster *owner, size_t index) { leaving_link_t *result; /* Structure à retourner */ result = malloc(sizeof(leaving_link_t)); result->owner = owner; result->index = index; result->straight = false; result->forced_straight = false; result->straight_level = SIZE_MAX; result->cluster_exit = false; return result; } /****************************************************************************** * * * Paramètres : link = structure à libérer de la mémoire. * * * * Description : Détruit un point d'attache pour un lien sortant. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void delete_leaving_link(leaving_link_t *link) { free(link); } /****************************************************************************** * * * Paramètres : link = information sur un lien à consulter. * * * * Description : Calcule l'abscisse d'un lien à son départ d'un bloc. * * * * Retour : Abscisse à attribuer à un départ de lien. * * * * Remarques : - * * * ******************************************************************************/ gint compute_leaving_link_position(const leaving_link_t *link) { gint result; /* Position à retourner */ result = g_graph_cluster_compute_leaving_link_position(link->owner, link->index); return result; } /****************************************************************************** * * * Paramètres : link = information sur un lien à consulter. * * x1 = abscisse de départ du lien d'origine. * * max = ordonnée la plus profonde à ne pas dépasser. * * * * Description : Détermine une direction prise par un lien à son départ. * * * * Retour : Direction prise à l'écran. * * * * Remarques : - * * * ******************************************************************************/ LeavingLinkDir get_leaving_link_direction(const leaving_link_t *link, gint x1, gint max) { LeavingLinkDir result; /* Préférence à retourner */ GGraphCluster *owner; /* Raccourci vers le proprio */ GtkAllocation alloc; /* Emplacement reservé */ gint x2; /* Abscisse d'arrivée de lien */ owner = link->other->owner; g_graph_cluster_get_allocation(owner, &alloc); if (alloc.y > max) result = LLD_NO_PREF; else { result = g_graph_cluster_get_link_direction(owner, x1, max); if (result == LLD_NO_PREF) { /** * Les liens ne sont pas encore ordonnés avec leur indice final. * Donc on choisit de faire au plus simple, et donc au plus rapide. * * Une alternative viable, mais tout aussi imprécise, serait d'appeler : * * idx = g_graph_cluster_find_incoming_link(owner, link); * * x2 = g_graph_cluster_compute_incoming_link_position(owner, idx); */ x2 = alloc.x + alloc.width / 2; if (x1 < x2) result = LLD_TO_RIGHT; else if (x1 > x2) result = LLD_TO_LEFT; else result = LLD_NO_PREF; } } return result; } /****************************************************************************** * * * Paramètres : a = premier lien entrant à comparer. * * b = second lien entrant à comparer. * * info = compléments d'information pour l'opération. * * * * Description : Compare deux liens sortants. * * * * Retour : Bilan de comparaison. * * * * Remarques : - * * * ******************************************************************************/ int cmp_leaving_links(const leaving_link_t **a, const leaving_link_t **b, const leaving_cmp_info_t *info) { int result; /* Bilan à retourner */ GGraphCluster *owner; /* Raccourci vers le proprio */ gint pos_a; /* Point de départ pour A */ GtkAllocation alloc; /* Emplacement de cluster */ gint pos_b; /* Point de départ pour B */ /* Calcul des ordonnées des points de chute */ owner = (*a)->other->owner; pos_a = G_MAXINT; if (!g_graph_cluster_compute_min_y_target((*a)->other->owner, info->root, &pos_a)) { g_graph_cluster_get_allocation(owner, &alloc); pos_a = alloc.y; } owner = (*b)->other->owner; pos_b = G_MAXINT; if (!g_graph_cluster_compute_min_y_target((*b)->other->owner, info->root, &pos_b)) { g_graph_cluster_get_allocation(owner, &alloc); pos_b = alloc.y; } /* Comparaison */ if (pos_a < pos_b) result = -1; else if (pos_a > pos_b) result = 1; else result = 0; if (info->dir == LLD_TO_RIGHT) result *= -1; return result; }