/* Chrysalide - Outil d'analyse de fichiers binaires * grid.c - composant d'affichage avec des chemins vers les composants contenus * * Copyright (C) 2018-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 "grid.h" #include "grid-int.h" /* -------------------------- GESTION DES TUILES AFFICHEES -------------------------- */ #define IS_LEAF_TILE(t) \ ({ \ bool __result; \ __result = GTK_IS_DOCK_STATION((t)->widget); \ assert(__result || GTK_IS_PANED((t)->widget)); \ __result; \ }) /* Supprime une tuile de la mémoire. */ static void delete_tile(grid_tile_t *); /* --------------------------- INTERFACE DU COMPOSANT GTK --------------------------- */ /* Initialise la classe des conteneurs d'affichage en tuiles. */ static void gtk_tiling_grid_class_init(GtkTilingGridClass *); /* Initialise une instance de conteneur d'affichage en tuiles. */ static void gtk_tiling_grid_init(GtkTilingGrid *); /* Supprime toutes les références externes. */ static void gtk_tiling_grid_dispose(GtkTilingGrid *); /* Procède à la libération totale de la mémoire. */ static void gtk_tiling_grid_finalize(GtkTilingGrid *); /* ---------------------------------------------------------------------------------- */ /* GESTION DES TUILES AFFICHEES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : tile = tuile à supprimer. * * * * Description : Supprime une tuile de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void delete_tile(grid_tile_t *tile) { if (!IS_LEAF_TILE(tile)) { delete_tile(tile->children[0]); delete_tile(tile->children[1]); } else free(tile->path); unref_object(tile->widget); free(tile); } /* ---------------------------------------------------------------------------------- */ /* INTERFACE DU COMPOSANT GTK */ /* ---------------------------------------------------------------------------------- */ /* Détermine le type du conteneur d'affichage en tuiles nommées. */ G_DEFINE_TYPE(GtkTilingGrid, gtk_tiling_grid, GTK_TYPE_WIDGET) /****************************************************************************** * * * Paramètres : klass = classe GTK à initialiser. * * * * Description : Initialise la classe des conteneurs d'affichage en tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_tiling_grid_class_init(GtkTilingGridClass *klass) { GObjectClass *object; /* Autre version de la classe */ object = G_OBJECT_CLASS(klass); object->dispose = (GObjectFinalizeFunc/* ! */)gtk_tiling_grid_dispose; object->finalize = (GObjectFinalizeFunc)gtk_tiling_grid_finalize; g_signal_new("station-created", GTK_TYPE_TILING_GRID, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET(GtkTilingGridClass, station_created), NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_WIDGET/*DOCK_STATION FIXME */); } /****************************************************************************** * * * Paramètres : tgrid = instance GTK à initialiser. * * * * Description : Initialise une instance de conteneur d'affichage en tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_tiling_grid_init(GtkTilingGrid *tgrid) { tgrid->tiles = NULL; tgrid->def_panel = NULL; } /****************************************************************************** * * * Paramètres : grid = instance d'objet GLib à traiter. * * * * Description : Supprime toutes les références externes. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_tiling_grid_dispose(GtkTilingGrid *grid) { if (grid->tiles != NULL) { delete_tile(grid->tiles); grid->tiles = NULL; } g_clear_object(&grid->def_panel); G_OBJECT_CLASS(gtk_tiling_grid_parent_class)->dispose(G_OBJECT(grid)); } /****************************************************************************** * * * Paramètres : grid = instance d'objet GLib à traiter. * * * * Description : Procède à la libération totale de la mémoire. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void gtk_tiling_grid_finalize(GtkTilingGrid *grid) { G_OBJECT_CLASS(gtk_tiling_grid_parent_class)->finalize(G_OBJECT(grid)); } /****************************************************************************** * * * Paramètres : - * * * * Description : Crée une nouvelle instance de conteneur avec tuiles. * * * * Retour : Composant GTK mis en place. * * * * Remarques : - * * * ******************************************************************************/ GtkWidget *gtk_tiling_grid_new(void) { GtkWidget *result; /* Instance à retourner */ result = g_object_new(GTK_TYPE_TILING_GRID, NULL); return result; } #if 0 #include <assert.h> #include <ctype.h> #include <malloc.h> #include <string.h> #include "../core/logs.h" /* Valide un chemin d'accès à une tuile. */ static bool is_valid_tile_path(const char *); /* Crée une tuile finale d'affichage de panneaux. */ static grid_tile_t *create_leaf_tile(const char *, GtkTiledGrid *); /* Crée une tuile intermédiaire d'affichage de panneaux. */ static grid_tile_t *create_inter_tile(grid_tile_t *, bool, grid_tile_t *, grid_tile_t *); /* Supprime une tuile de la mémoire. */ //static void delete_tile(grid_tile_t *); /* Calcule la taille comme entre un chemin et celui d'une tuile. */ static size_t compute_tile_score(const grid_tile_t *, const char *); /* Indique la tuile adaptée pour un chemin donné. */ static grid_tile_t *find_suitable_tile(grid_tile_t **, const char *, GtkTiledGrid *); /* Découpe une tuile pour y insérer une zone. */ static grid_tile_t *split_tile(grid_tile_t **, const char *, char, GtkTiledGrid *); /* Tente de mettre la main sur une station d'accueil. */ static grid_tile_t *find_tile_for_widget(grid_tile_t *, GtkWidget *); /* Retire une moitié de tuile vide au plein profit de l'autre. */ static void collapse_tile(grid_tile_t *, grid_tile_t *); /* --------------------------- INTERFACE DU COMPOSANT GTK --------------------------- */ /* ---------------------------------------------------------------------------------- */ /* GESTION DES TUILES AFFICHEES */ /* ---------------------------------------------------------------------------------- */ /****************************************************************************** * * * Paramètres : path = chemin destiné à sélectionner une tuile. * * * * Description : Valide un chemin d'accès à une tuile. * * * * Retour : true si le chemin est utilisable, false sinon. * * * * Remarques : - * * * ******************************************************************************/ static bool is_valid_tile_path(const char *path) { bool result; /* Bilan à retourner */ size_t len; /* Taille du chemin */ size_t i; /* Boucle de parcours */ char c; /* Caractère de chemin analysé */ /** * M[NESWnesw]* */ len = strlen(path); result = (len >= 1); if (result) result = (path[0] == 'M'); for (i = 1; i < len && result; i++) { c = path[i]; if (c == '\0') break; result = (c == 'N' || c == 'n' || c == 'E' || c == 'e' || c == 'S' || c == 's' || c == 'W' || c == 'w'); } return result; } /****************************************************************************** * * * Paramètres : path = chemin d'accès à la future tuile. * * tgrid = conteneur d'affichage en tuiles à manipuler. * * * * Description : Crée une tuile finale d'affichage de panneaux. * * * * Retour : Structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ static grid_tile_t *create_leaf_tile(const char *path, GtkTiledGrid *tgrid) { grid_tile_t *result; /* Structure à retourner */ result = (grid_tile_t *)malloc(sizeof(grid_tile_t)); result->parent = NULL; result->widget = gtk_dock_station_new(); gtk_widget_show(result->widget); result->path = strdup(path); result->children[0] = NULL; result->children[1] = NULL; g_signal_emit_by_name(tgrid, "station-created", result->widget); return result; } /****************************************************************************** * * * Paramètres : parent = tuile parente ou NULL si aucune. * * horiz = indique le type d'orientation désiré. * * first = première tuile à intégrer. * * second = seconde tuile à intégrer. * * * * Description : Crée une tuile intermédiaire d'affichage de panneaux. * * * * Retour : Structure mise en place. * * * * Remarques : - * * * ******************************************************************************/ static grid_tile_t *create_inter_tile(grid_tile_t *parent, bool horiz, grid_tile_t *first, grid_tile_t *second) { grid_tile_t *result; /* Structure à retourner */ GtkWidget *container; /* Conteneur à vider */ result = (grid_tile_t *)malloc(sizeof(grid_tile_t)); result->parent = parent; if (horiz) result->widget = gtk_paned_new(GTK_ORIENTATION_HORIZONTAL); else result->widget = gtk_paned_new(GTK_ORIENTATION_VERTICAL); gtk_widget_show(result->widget); result->path = NULL; result->children[0] = first; result->children[1] = second; /* Changement de propriétaire */ container = gtk_widget_get_parent(first->widget); if (container != NULL) gtk_container_remove(GTK_CONTAINER(container), first->widget); g_object_ref(G_OBJECT(first->widget)); gtk_paned_pack1(GTK_PANED(result->widget), first->widget, TRUE, FALSE); container = gtk_widget_get_parent(second->widget); if (container != NULL) gtk_container_remove(GTK_CONTAINER(container), second->widget); g_object_ref(G_OBJECT(second->widget)); gtk_paned_pack2(GTK_PANED(result->widget), second->widget, TRUE, FALSE); return result; } /****************************************************************************** * * * Paramètres : tile = tuile à analyser. * * path = chemin final complet recherché. * * * * Description : Calcule la taille comme entre un chemin et celui d'une tuile.* * * * Retour : Quantité de caractères communs. * * * * Remarques : - * * * ******************************************************************************/ static size_t compute_tile_score(const grid_tile_t *tile, const char *path) { size_t result; /* Nombre de points à renvoyer */ size_t max; /* Taille du chemin de la tuile*/ size_t i; /* Boucle de parcours */ size_t score_0; /* Score du sous-élément #1 */ size_t score_1; /* Score du sous-élément #2 */ if (IS_LEAF_TILE(tile)) { max = strlen(tile->path); if (strlen(path) < max) result = 0; else { result = 0; for (i = 0; i < max; i++) { if (tolower((unsigned char)tile->path[i]) == tolower((unsigned char)path[i])) result++; else break; } } } else { score_0 = compute_tile_score(tile->children[0], path); score_1 = compute_tile_score(tile->children[1], path); result = score_0 > score_1 ? score_0 : score_1; } return result; } /****************************************************************************** * * * Paramètres : tile = tuile ou NULL si aucune. [OUT] * * path = chemin d'accès à la tuile visée. * * tgrid = conteneur d'affichage en tuiles à manipuler. * * * * Description : Indique la tuile adaptée pour un chemin donné. * * * * Retour : Structure d'acceuil à disposition. * * * * Remarques : - * * * ******************************************************************************/ static grid_tile_t *find_suitable_tile(grid_tile_t **tile, const char *path, GtkTiledGrid *tgrid) { grid_tile_t *result; /* Structure à renvoyer */ size_t best_len; /* Taille du chemin associé */ size_t score_0; /* Score du sous-élément #1 */ size_t score_1; /* Score du sous-élément #2 */ char *sub_path; /* Nouvelle tentative d'accès */ grid_tile_t **best; /* Direction à prendre */ unsigned char next; /* Prochaine étape */ /* Cas d'école : appel initial */ if (*tile == NULL) { assert(path[0] == 'M' && path[1] == '\0'); result = create_leaf_tile("M", tgrid); *tile = result; } else { if (IS_LEAF_TILE(*tile)) { best_len = compute_tile_score(*tile, path); assert(best_len > 0); if (path[best_len] == '\0') result = *tile; else result = split_tile(tile, path, path[best_len], tgrid); } else { score_0 = compute_tile_score((*tile)->children[0], path); score_1 = compute_tile_score((*tile)->children[1], path); assert(score_0 > 0 || score_0 > 0); if (score_0 == score_1) { sub_path = strndup(path, score_0); score_0 = compute_tile_score((*tile)->children[0], sub_path); score_1 = compute_tile_score((*tile)->children[1], sub_path); free(sub_path); } if (score_0 == score_1) result = split_tile(tile, path, path[score_0], tgrid); else { if (score_0 > score_1) { best = &(*tile)->children[0]; best_len = score_0; } else { best = &(*tile)->children[1]; best_len = score_1; } /** * Si on vient de tomber une feuille, trois cas de figure : * - soit c'est elle qui est visée. * - soit on veut la diviser. * - soit on veut la diviser en englobant ses voisines. */ if (IS_LEAF_TILE(*best)) { assert(best_len <= strlen(path)); next = path[best_len]; /* Premier cas */ if (next == '\0') result = *best; else { /* Second cas */ if (islower(next)) result = find_suitable_tile(best, path, tgrid); /* Troisième cas */ else result = split_tile(tile, path, next, tgrid); } } else result = find_suitable_tile(best, path, tgrid); } } } assert(IS_LEAF_TILE(result)); return result; } /****************************************************************************** * * * Paramètres : tile = tuile à découper en deux. [OUT] * * path = chemin d'accès à la future tuile. * * endpoint = désignation de la zone représentée. * * tgrid = conteneur d'affichage en tuiles à manipuler. * * * * Description : Découpe une tuile pour y insérer une zone. * * * * Retour : Structure fille mise en place. * * * * Remarques : - * * * ******************************************************************************/ static grid_tile_t *split_tile(grid_tile_t **tile, const char *path, char endpoint, GtkTiledGrid *tgrid) { grid_tile_t *result; /* Création à retourner */ GtkWidget *container; /* Conteneur à vider */ grid_tile_t *new; /* Nouvelle tuile intermédiaire*/ container = gtk_widget_get_parent((*tile)->widget); /* Création */ result = create_leaf_tile(path, tgrid); /* Encapsulation */ switch (endpoint) { case 'N': case 'n': new = create_inter_tile((*tile)->parent, false, result, *tile); break; case 'E': case 'e': new = create_inter_tile((*tile)->parent, true, *tile, result); break; case 'S': case 's': new = create_inter_tile((*tile)->parent, false, *tile, result); break; case 'W': case 'w': new = create_inter_tile((*tile)->parent, true, result, *tile); break; default: assert(false); new = NULL; break; } /* Connexions */ *tile = new; result->parent = new; if (container != NULL) { g_object_ref(G_OBJECT(new->widget)); gtk_container_add(GTK_CONTAINER(container), new->widget); } return result; } /****************************************************************************** * * * Paramètres : tile = tuile parente, prochaine victime de promotion. * * side = côté de tuile amené à disparaître. * * * * Description : Retire une moitié de tuile vide au plein profit de l'autre. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ static void collapse_tile(grid_tile_t *tile, grid_tile_t *side) { grid_tile_t *promoted; /* Tuile à faire remonter */ GtkWidget *container; /* Conteneur à vider */ assert(!IS_LEAF_TILE(tile)); /* Sélection du remplaçant */ if (side == tile->children[0]) promoted = tile->children[1]; else promoted = tile->children[0]; /* Etablissement d'une place nette */ gtk_container_remove(GTK_CONTAINER(tile->widget), promoted->widget); container = gtk_widget_get_parent(tile->widget); gtk_container_remove(GTK_CONTAINER(container), tile->widget); delete_tile(side); /* Promotion effective */ tile->widget = promoted->widget; tile->path = promoted->path; tile->children[0] = promoted->children[0]; tile->children[1] = promoted->children[1]; g_object_ref(G_OBJECT(promoted->widget)); gtk_container_add(GTK_CONTAINER(container), tile->widget); free(promoted); } /****************************************************************************** * * * Paramètres : tile = point de départ des recherches locales. * * widget = composant graphique à retrouver. * * * * Description : Tente de mettre la main sur une station d'accueil. * * * * Retour : Eventuelle tuile trouvée ou NULL. * * * * Remarques : - * * * ******************************************************************************/ static grid_tile_t *find_tile_for_widget(grid_tile_t *tile, GtkWidget *widget) { grid_tile_t *result; /* Tuile à retourner */ if (IS_LEAF_TILE(tile)) result = tile->widget == widget ? tile : NULL; else { result = find_tile_for_widget(tile->children[0], widget); if (result == NULL) result = find_tile_for_widget(tile->children[1], widget); } return result; } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à consulter. * * * * Description : Donne le panneau fourni par défaut pour la zone principale. * * * * Retour : Panneau d'affichage par défault ou NULL. * * * * Remarques : - * * * ******************************************************************************/ GPanelItem *gtk_tiled_grid_get_default_main_panel(const GtkTiledGrid *tgrid) { GPanelItem *result; /* Panneau à retourner */ result = tgrid->def_panel; if (result != NULL) g_object_ref(G_OBJECT(result)); return result; } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à modifier. * * panel = panneau d'affichage par défault ou NULL. * * * * Description : Fournit le panneau par défaut pour la zone principale. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_tiled_grid_set_default_main_panel(GtkTiledGrid *tgrid, GPanelItem *panel) { GtkWidget *widget; /* Composant GTK à retirer */ GtkWidget *parent; /* Conteneur à vider */ grid_tile_t *tile; /* Première tuile d'accueil */ if (tgrid->def_panel != NULL) { widget = gtk_dockable_build_widget(GTK_DOCKABLE(tgrid->def_panel)); parent = gtk_widget_get_parent(widget); if (parent != NULL) gtk_container_remove(GTK_CONTAINER(parent), widget); g_object_unref(G_OBJECT(widget)); g_object_unref(G_OBJECT(tgrid->def_panel)); } tgrid->def_panel = panel; if (panel != NULL) { g_object_ref(G_OBJECT(panel)); if (tgrid->tiles == NULL) gtk_tiled_grid_add(tgrid, panel); else { tile = find_suitable_tile(&tgrid->tiles, "M", tgrid); if (gtk_notebook_get_n_pages(GTK_NOTEBOOK(tile->widget)) == 0) gtk_tiled_grid_add(tgrid, panel); } } } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à modifier. * * panel = panneau d'affichage à intégrer. * * * * Description : Incorpore un nouveau panneau dans le conteneur en tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_tiled_grid_add(GtkTiledGrid *tgrid, GPanelItem *panel) { char *path; /* Chemin d'accès */ char *name; /* Nom à donner à l'onglet */ grid_tile_t *tile; /* Tuile d'accueil */ path = gtk_panel_item_class_get_path(G_PANEL_ITEM_GET_CLASS(panel)); if (!is_valid_tile_path(path)) { name = gtk_dockable_get_name(GTK_DOCKABLE(panel)); log_variadic_message(LMT_ERROR, _("Invalid path '%s' for panel '%s'"), path, name); free(name); } else { tile = find_suitable_tile(&tgrid->tiles, path, tgrid); assert(tile != NULL); gtk_dock_station_add_dockable(GTK_DOCK_STATION(tile->widget), GTK_DOCKABLE(panel)); g_panel_item_set_dock_at_startup(panel, true); /* Si c'est la toute première fois... */ if (gtk_widget_get_parent(tile->widget) == NULL) { assert(tile == tgrid->tiles); assert(tile->path[0] == 'M' && tile->path[1] == '\0'); g_object_ref(G_OBJECT(tile->widget)); gtk_container_add(GTK_CONTAINER(tgrid), tile->widget); } /* Si on n'a plus besoin du panneau par défaut */ if (tgrid->def_panel != NULL && tile->path[0] == 'M' && tile->path[1] == '\0') { /* Si ce n'est pas le panneau qu'on vient de rajouter...*/ if (panel != tgrid->def_panel) { /* Enfin : si ce panneau par défaut est réellement en place */ if (g_panel_item_is_docked(tgrid->def_panel)) gtk_tiled_grid_remove(tgrid, tgrid->def_panel); } } } free(path); } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à modifier. * * panel = panneau d'affichage à supprimer. * * * * Description : Retire un panneau dans le conteneur en tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_tiled_grid_remove(GtkTiledGrid *tgrid, GPanelItem *panel) { GtkWidget *station; /* Support courant */ grid_tile_t *tile; /* Tuile d'accueil */ assert(g_panel_item_is_docked(panel)); gtk_dockable_decompose(GTK_DOCKABLE(panel), &station); tile = find_tile_for_widget(tgrid->tiles, station); assert(tile != NULL); gtk_dock_station_remove_dockable(GTK_DOCK_STATION(station), GTK_DOCKABLE(panel)); g_panel_item_set_dock_at_startup(panel, false); if (gtk_notebook_get_n_pages(GTK_NOTEBOOK(station)) == 0) { /* Si le panneau par défaut devient nécessaire */ if (tgrid->def_panel != NULL && tile->path[0] == 'M' && tile->path[1] == '\0') gtk_tiled_grid_add(tgrid, tgrid->def_panel); else { /* La racine est concernée ! */ if (tile->parent == NULL) { assert(tile == tgrid->tiles); g_object_ref(G_OBJECT(tile->widget)); gtk_container_remove(GTK_CONTAINER(tgrid), tile->widget); delete_tile(tile); tgrid->tiles = NULL; } else collapse_tile(tile->parent, tile); } } } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à consulter. * * station = station d'accueil à retrouver. * * * * Description : Indique le chemin correspondant à une station intégrée. * * * * Retour : Copie de chemin trouvé, à libérer ensuite, ou NULL si échec. * * * * Remarques : - * * * ******************************************************************************/ char *gtk_tiled_grid_get_path_for_station(const GtkTiledGrid *tgrid, GtkDockStation *station) { char *result; /* Chemin d'accès à renvoyer */ grid_tile_t *tile; /* Tuile d'accueil */ tile = find_tile_for_widget(tgrid->tiles, GTK_WIDGET(station)); if (tile == NULL) result = NULL; else result = strdup(tile->path); return result; } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à mettre à jour. * * config = configuration à consulter. * * * * Description : Replace les positions des séparateurs de tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_tiled_grid_restore_positions(const GtkTiledGrid *tgrid, GGenConfig *config) { void visit_tiles_for_restoring(grid_tile_t *tile, const char *vpath) { GtkOrientation orientation; /* Direction de la tuile */ char hint; /* Inutile donc indispensable */ char *key; /* Clef d'accès à un paramètre */ gint position; /* Nouvelle position de barre */ size_t i; /* Boucle de parcours */ char *child_key; /* Clef d'accès des suivants */ if (!IS_LEAF_TILE(tile)) { orientation = gtk_orientable_get_orientation(GTK_ORIENTABLE(tile->widget)); hint = orientation == GTK_ORIENTATION_HORIZONTAL ? 'h' : 'v'; asprintf(&key, "%s%c", vpath, hint); if (g_generic_config_get_value(config, key, &position)) gtk_paned_set_position(GTK_PANED(tile->widget), position); for (i = 0; i < 2; i++) { asprintf(&child_key, "%s%zu", key, i); visit_tiles_for_restoring(tile->children[i], child_key); free(child_key); } free(key); } } visit_tiles_for_restoring(tgrid->tiles, "gui.panels.positions.R"); } /****************************************************************************** * * * Paramètres : tgrid = conteneur d'affichage en tuiles à consulter. * * config = configuration à mettre à jour. * * * * Description : Sauvegarde les positions des séparateurs de tuiles. * * * * Retour : - * * * * Remarques : - * * * ******************************************************************************/ void gtk_tiled_grid_save_positions(const GtkTiledGrid *tgrid, GGenConfig *config) { void visit_tiles_for_saving(grid_tile_t *tile, const char *vpath) { GtkOrientation orientation; /* Direction de la tuile */ char hint; /* Inutile donc indispensable */ char *key; /* Clef d'accès à un paramètre */ gint position; /* Nouvelle position de barre */ size_t i; /* Boucle de parcours */ char *child_key; /* Clef d'accès des suivants */ if (!IS_LEAF_TILE(tile)) { orientation = gtk_orientable_get_orientation(GTK_ORIENTABLE(tile->widget)); hint = orientation == GTK_ORIENTATION_HORIZONTAL ? 'h' : 'v'; asprintf(&key, "%s%c", vpath, hint); position = gtk_paned_get_position(GTK_PANED(tile->widget)); g_generic_config_create_or_udpdate_param(config, key, CPT_INTEGER, -1, position); for (i = 0; i < 2; i++) { asprintf(&child_key, "%s%zu", key, i); visit_tiles_for_saving(tile->children[i], child_key); free(child_key); } free(key); } } visit_tiles_for_saving(tgrid->tiles, "gui.panels.positions.R"); } #endif