2024-06-30 17:18:18 +02:00
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
|
|
* *
|
2024-07-08 07:52:27 +02:00
|
|
|
* Gem-graph client *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
2024-07-21 23:17:01 +02:00
|
|
|
* Signals *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
2024-07-08 07:52:27 +02:00
|
|
|
* Copyright © 2021 Libre en Communs <contact@a-lec.org> *
|
|
|
|
* Copyright © 2021 Adrien Bourmault <neox@a-lec.org> *
|
|
|
|
* Copyright © 2021 Jean Sirmai <jean@a-lec.org> *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
|
|
|
* This file is part of Gem-graph. *
|
|
|
|
* *
|
2024-07-08 07:52:27 +02:00
|
|
|
* This program is free software: you can redistribute it and/or modify it *
|
|
|
|
* under the terms of the GNU Affero General Public License *
|
|
|
|
* as published by the Free Software Foundation, *
|
|
|
|
* either version 3 of the License, *
|
|
|
|
* or (at your option) any later version. *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
2024-07-08 07:52:27 +02:00
|
|
|
* This program 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 Affero General Public License for more details. *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
2024-07-08 07:52:27 +02:00
|
|
|
* You should have received a copy of the GNU Affero General Public License *
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>. *
|
2024-06-30 17:18:18 +02:00
|
|
|
* *
|
|
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
|
|
|
|
|
2024-07-08 07:52:27 +02:00
|
|
|
|
2024-07-18 22:06:24 +02:00
|
|
|
#include "../include/fsm.h"
|
|
|
|
#include "../include/signal.h"
|
|
|
|
#include "../include/widget.h"
|
2024-07-23 19:43:21 +02:00
|
|
|
#include "../include/prefer.h"
|
2024-07-18 22:06:24 +02:00
|
|
|
#include "../include/graphics.h"
|
2024-04-22 18:40:30 +02:00
|
|
|
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-07-06 08:27:05 +02:00
|
|
|
static void on_auto_notification (const char *message)
|
|
|
|
{
|
|
|
|
/* Ignored (2024-06-06) because I don't know how to get "main_window" easily
|
|
|
|
|
|
|
|
if (window->toast_revealer == NULL) {
|
|
|
|
g_printerr("Can't find app->toast_overlay !\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->toast_text == NULL) {
|
|
|
|
g_printerr("Can't find app->toast_overlay !\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_label_set_label(window->toast_text, message);
|
|
|
|
gtk_revealer_set_reveal_child(window->toast_revealer, true);
|
|
|
|
*/
|
|
|
|
g_printerr("%s\n", message);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-04-22 18:40:30 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
/* T R E E */
|
|
|
|
/******************************************************************************/
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-07-08 09:13:11 +02:00
|
|
|
static void on_user_tree_expander_toggled (GtkExpander *expander,
|
2024-06-30 11:53:03 +02:00
|
|
|
gpointer user_data)
|
2024-04-22 18:40:30 +02:00
|
|
|
{
|
|
|
|
GtkTreeListRow *row = GTK_TREE_LIST_ROW (user_data);
|
2024-06-30 11:53:03 +02:00
|
|
|
gboolean is_expanded = gtk_tree_list_row_get_expanded (row);
|
|
|
|
gtk_tree_list_row_set_expanded (row,
|
2024-07-08 07:52:27 +02:00
|
|
|
! is_expanded);
|
2024-04-22 18:40:30 +02:00
|
|
|
}
|
|
|
|
|
2024-06-30 11:53:03 +02:00
|
|
|
|
|
|
|
void on_bind_user_tree_factory (GtkSignalListItemFactory *factory,
|
|
|
|
GObject* object,
|
|
|
|
gpointer user_data)
|
2024-04-22 18:40:30 +02:00
|
|
|
{
|
2024-06-30 11:53:03 +02:00
|
|
|
GtkListItem *list_item = GTK_LIST_ITEM (object);
|
|
|
|
assert (list_item);
|
|
|
|
|
|
|
|
GtkTreeListRow *row = gtk_list_item_get_item (list_item);
|
2024-07-08 07:52:27 +02:00
|
|
|
assert (row); // if (row != NULL) {...} do something ? TODO Check !
|
2024-06-30 11:53:03 +02:00
|
|
|
|
|
|
|
const gchar *text = gtk_string_object_get_string (
|
|
|
|
GTK_STRING_OBJECT (gtk_tree_list_row_get_item (row)));
|
2024-04-22 18:40:30 +02:00
|
|
|
GtkWidget *expander = gtk_list_item_get_child (list_item);
|
|
|
|
gtk_expander_set_label (GTK_EXPANDER (expander), text);
|
2024-06-30 11:53:03 +02:00
|
|
|
g_signal_handlers_disconnect_by_func (expander,
|
|
|
|
G_CALLBACK (on_user_tree_expander_toggled),
|
|
|
|
row);
|
|
|
|
g_signal_connect (expander,
|
|
|
|
"activate",
|
|
|
|
G_CALLBACK (on_user_tree_expander_toggled),
|
|
|
|
row);
|
|
|
|
gtk_widget_set_margin_start (expander,
|
|
|
|
gtk_tree_list_row_get_depth(row) * 20);
|
2024-04-22 18:40:30 +02:00
|
|
|
}
|
|
|
|
|
2024-07-01 07:26:16 +02:00
|
|
|
void on_setup_user_tree_factory (GtkSignalListItemFactory *factory,
|
|
|
|
GObject* object, gpointer user_data){
|
|
|
|
GtkWidget* expander = gtk_expander_new (NULL);
|
|
|
|
gtk_list_item_set_child (GTK_LIST_ITEM (object), expander);
|
|
|
|
if (0) printf("[on_setup_user_tree_factory] here is an expander\n");
|
|
|
|
}
|
|
|
|
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-04-22 18:40:30 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
/* G L A R E A */
|
|
|
|
/******************************************************************************/
|
2024-06-30 11:53:03 +02:00
|
|
|
|
|
|
|
gboolean on_glarea_render(GtkGLArea *area,
|
|
|
|
GdkGLContext *context)
|
2024-04-22 18:40:30 +02:00
|
|
|
{
|
|
|
|
// Check if the widget is a glarea
|
|
|
|
if(gtk_gl_area_get_error(area) != NULL) {
|
|
|
|
on_auto_notification("An OpenGL error occured !");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-07-18 23:51:04 +02:00
|
|
|
if (graphics_render_stack(gtk_widget_get_parent(GTK_WIDGET(area))) == false) {
|
2024-04-22 18:40:30 +02:00
|
|
|
on_auto_notification("Failed to render corresponding graphic stack !");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* We need to set up our state when we realize the GtkGLArea widget */
|
|
|
|
void on_glarea_realize(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
GError *internal_error = NULL;
|
|
|
|
|
|
|
|
// Make the GL context current to be able to call the GL API
|
|
|
|
gtk_gl_area_make_current(GTK_GL_AREA(widget));
|
|
|
|
|
|
|
|
// Check if the widget is a glarea
|
|
|
|
if(gtk_gl_area_get_error(GTK_GL_AREA(widget)) != NULL) {
|
|
|
|
on_auto_notification("An OpenGL error occured !");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Link graphical stack to widget
|
2024-07-18 23:51:04 +02:00
|
|
|
if (graphics_init_graphics_stack(gtk_widget_get_parent(widget),
|
2024-04-22 18:40:30 +02:00
|
|
|
internal_error) == false) {
|
|
|
|
on_auto_notification(
|
|
|
|
"Failed to link the graphic stack to widgets !");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_gl_area_set_auto_render(GTK_GL_AREA(widget), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We should tear down the state when unrealizing */
|
|
|
|
void on_glarea_unrealize(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
GError *internal_error = NULL;
|
|
|
|
|
|
|
|
// Make the GL context current to be able to call the GL API
|
|
|
|
gtk_gl_area_make_current(GTK_GL_AREA(widget));
|
|
|
|
|
|
|
|
// Check if the widget is a glarea
|
|
|
|
if(gtk_gl_area_get_error(GTK_GL_AREA(widget)) != NULL) {
|
|
|
|
on_auto_notification("An OpenGL error occured !");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroy graphic stack
|
2024-07-18 23:51:04 +02:00
|
|
|
if (graphics_shutdown_graphics_stack(gtk_widget_get_parent(widget),
|
2024-04-22 18:40:30 +02:00
|
|
|
internal_error) == false) {
|
|
|
|
on_auto_notification(
|
|
|
|
"Failed to shutdown the graphic stack !");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void on_axis_value_change(GtkAdjustment *adjustment, gpointer data)
|
|
|
|
{
|
|
|
|
|
|
|
|
GtkWidget *slider = gtk_widget_get_parent(GTK_WIDGET(data));
|
|
|
|
GtkWidget *container_widget = gtk_widget_get_parent(GTK_WIDGET(slider));
|
|
|
|
|
|
|
|
const gchar *label_text = gtk_label_get_label(GTK_LABEL(data));
|
|
|
|
|
|
|
|
// THANKS ASCIIIII/Unicode/Whateverrr !
|
|
|
|
int axis = label_text[0] - 'X';
|
|
|
|
|
|
|
|
g_assert(axis >= 0 && axis < N_AXIS);
|
|
|
|
|
|
|
|
/* Update the rotation angle */
|
2024-07-18 23:51:04 +02:00
|
|
|
graphics_update_axis_stack(container_widget,
|
2024-04-22 18:40:30 +02:00
|
|
|
axis,
|
|
|
|
gtk_adjustment_get_value(adjustment));
|
|
|
|
|
|
|
|
/* Update the contents of the GL drawing area */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2024-07-08 07:52:27 +02:00
|
|
|
/* D I A L O G W I D G E T S */
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2024-07-21 13:01:30 +02:00
|
|
|
void on_save_current_model_before_editing (GtkWidget *btt_SAVE_CURRENT_MODEL,
|
2024-07-08 07:52:27 +02:00
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
gtk_widget_set_sensitive (GTK_WIDGET (data),
|
|
|
|
TRUE);
|
2024-07-21 23:17:01 +02:00
|
|
|
printf ("signal.c - SAVE_CURRENT_MODEL_BEFORE_EDITING\n");
|
2024-07-08 07:52:27 +02:00
|
|
|
}
|
|
|
|
|
2024-07-21 13:01:30 +02:00
|
|
|
void on_discard_current_model_before_editing (GtkWidget *btt_SAVE_CURRENT_MODEL,
|
2024-07-08 07:52:27 +02:00
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
gtk_window_close (GTK_WINDOW (data));
|
2024-07-21 23:17:01 +02:00
|
|
|
printf ("signal.c - DISCARD_CURRENT_MODEL_AND_START_EDITING\n");
|
2024-07-08 07:52:27 +02:00
|
|
|
}
|
|
|
|
|
2024-07-21 13:01:30 +02:00
|
|
|
void on_write_current_model (GtkWidget *btt_WRITE_CURRENT_MODEL,
|
2024-07-08 07:52:27 +02:00
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
gtk_window_close (GTK_WINDOW (data));
|
2024-07-21 23:17:01 +02:00
|
|
|
printf ("signal.c - WRITE_CURRENT_MODEL\n");
|
2024-07-08 07:52:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2024-07-25 05:55:15 +02:00
|
|
|
/* T O P B A R W I D G E T S */
|
2024-04-22 18:40:30 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
// ----------------------- E X E C / E D I T -------------------------- //
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
static void switch_state_rules_data();
|
2024-06-29 23:31:17 +02:00
|
|
|
|
2024-07-21 12:33:29 +02:00
|
|
|
void on_toggle_exec_edit (GtkWidget *toggled_button, gpointer user_data)
|
2024-07-06 08:27:05 +02:00
|
|
|
{
|
2024-07-21 12:33:29 +02:00
|
|
|
if (fsm_get_exec_edit ()) {
|
2024-06-30 11:53:03 +02:00
|
|
|
|
|
|
|
gtk_button_set_icon_name (GTK_BUTTON (toggled_button),
|
|
|
|
"power-profile-balanced-rtl-symbolic");
|
2024-07-21 12:33:29 +02:00
|
|
|
fsm_set_exec_edit (EXEC);
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-04-22 18:40:30 +02:00
|
|
|
} else {
|
2024-06-30 11:53:03 +02:00
|
|
|
|
|
|
|
gtk_button_set_icon_name (GTK_BUTTON (toggled_button),
|
|
|
|
"text-editor-symbolic");
|
|
|
|
|
|
|
|
// https://docs.gtk.org/gtk4/class.Window.html TODO 2024-06-30
|
2024-07-21 12:33:29 +02:00
|
|
|
// gtk_window_present (GTK_WINDOW (widget_get_dialog_window())); // works once only !
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-07-21 12:33:29 +02:00
|
|
|
fsm_set_exec_edit (EDIT);
|
2024-04-22 18:40:30 +02:00
|
|
|
}
|
2024-06-29 23:31:17 +02:00
|
|
|
|
2024-07-21 12:33:29 +02:00
|
|
|
switch_state_rules_data();
|
2024-04-22 18:40:30 +02:00
|
|
|
}
|
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
// ------------------ S T A T E / R U L E S / D A T A --------------- //
|
|
|
|
|
|
|
|
static void switch_state_rules_data()
|
|
|
|
{
|
|
|
|
switch (fsm_get_state_rules_data()) {
|
|
|
|
|
|
|
|
case (STATE) :
|
|
|
|
gtk_window_set_child (GTK_WINDOW (widget_get_main_window ()),
|
|
|
|
GTK_WIDGET (widget_get_state_page()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (RULES) :
|
|
|
|
gtk_window_set_child (GTK_WINDOW (widget_get_main_window ()),
|
|
|
|
GTK_WIDGET (widget_get_rules_page()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case (DATA) :
|
|
|
|
gtk_window_set_child (GTK_WINDOW (widget_get_main_window ()),
|
|
|
|
GTK_WIDGET (widget_get_stock_page()));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default :
|
|
|
|
printf("default in signal.on_toggle_state_rule_data()\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-21 23:17:01 +02:00
|
|
|
// TODO Il doit y avoir plus élégant,...
|
2024-07-21 12:33:29 +02:00
|
|
|
void on_toggle_state_rules_data (GtkWidget *toggled_button, gpointer user_data)
|
2024-07-06 08:27:05 +02:00
|
|
|
{
|
2024-06-30 11:53:03 +02:00
|
|
|
const char *toggled_button_name
|
|
|
|
= gtk_check_button_get_label (GTK_CHECK_BUTTON (toggled_button));
|
2024-04-22 18:40:30 +02:00
|
|
|
int is_active = gtk_check_button_get_active (GTK_CHECK_BUTTON (toggled_button));
|
|
|
|
|
2024-07-24 14:14:05 +02:00
|
|
|
if ( ! strcmp (toggled_button_name, "state")) fsm_set_state_rules_data (STATE);
|
|
|
|
if ( ! strcmp (toggled_button_name, "rules")) fsm_set_state_rules_data (RULES);
|
|
|
|
if ( ! strcmp (toggled_button_name, "data analysis")) fsm_set_state_rules_data (DATA);
|
2024-06-30 11:53:03 +02:00
|
|
|
|
2024-07-24 14:14:05 +02:00
|
|
|
if (is_active) switch_state_rules_data();
|
2024-04-22 18:40:30 +02:00
|
|
|
}
|
2024-07-05 22:19:18 +02:00
|
|
|
|
2024-07-23 00:17:35 +02:00
|
|
|
|
2024-07-23 21:08:18 +02:00
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
// ------------ M E N U / S E A R C H / H O M E / E T C ----------- //
|
2024-07-08 07:52:27 +02:00
|
|
|
|
2024-07-15 07:11:30 +02:00
|
|
|
void on_clicked_search (GtkWidget *btt_menu, gpointer list_box) {
|
2024-07-24 14:14:05 +02:00
|
|
|
// next line presents the text_window and works only once.\n
|
|
|
|
// It should present a menu.\n"); // TODO
|
2024-07-21 12:33:29 +02:00
|
|
|
gtk_window_present (GTK_WINDOW (widget_get_text_window()));
|
2024-07-15 07:11:30 +02:00
|
|
|
}
|
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
|
2024-07-21 13:01:30 +02:00
|
|
|
void on_clicked_home (GtkWidget *btt_reset, gpointer data)
|
2024-07-15 07:11:30 +02:00
|
|
|
{
|
2024-07-21 23:17:01 +02:00
|
|
|
printf ("signal.on_clicked_HOME() button presents the dialog_window\
|
2024-07-15 07:11:30 +02:00
|
|
|
( :- ) but it works only once.\n"); // TODO
|
2024-07-21 12:33:29 +02:00
|
|
|
gtk_window_present (GTK_WINDOW (widget_get_dialog_window()));
|
2024-07-15 07:11:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
|
2024-07-15 07:11:30 +02:00
|
|
|
/******************************************************************************/
|
2024-07-25 05:55:15 +02:00
|
|
|
/* S T A T E W I D G E T S */
|
2024-07-15 07:11:30 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2024-07-25 05:55:15 +02:00
|
|
|
// ---------------------------- O B J E C T S --------------------------- //
|
|
|
|
|
2024-07-25 19:59:29 +02:00
|
|
|
void on_updating_objects_transparencies (GtkWidget *btt_source, GtkScrollbar *scrollbar)
|
2024-07-25 05:55:15 +02:00
|
|
|
{
|
|
|
|
const char *btt_name = gtk_button_get_icon_name (GTK_BUTTON (btt_source));
|
2024-07-25 19:59:29 +02:00
|
|
|
int value = gtk_adjustment_get_value (gtk_scrollbar_get_adjustment (scrollbar));
|
2024-07-25 05:55:15 +02:00
|
|
|
|
2024-07-26 12:26:24 +02:00
|
|
|
if ( ! strcmp (btt_name, "document-revert-rtl-symbolic")) pref_set_store_restore_reset (STORE, value);
|
|
|
|
if ( ! strcmp (btt_name, "edit-undo-symbolic")) pref_set_store_restore_reset (RESTORE, value);
|
|
|
|
if ( ! strcmp (btt_name, "view-refresh-symbolic")) pref_set_store_restore_reset (RESET, value);
|
2024-07-25 23:25:04 +02:00
|
|
|
|
|
|
|
fsm_set_preferences_state (TRUE); // << à détailler TODO
|
2024-07-25 05:55:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// -------------------------- S IT U A T I O N S ------------------------ //
|
|
|
|
|
|
|
|
void on_situations_box_do_reset (GtkWidget *btt_reset, GtkScrollbar *reset_scrollbar)
|
|
|
|
{
|
|
|
|
GtkAdjustment *adj_situ = gtk_scrollbar_get_adjustment (reset_scrollbar);
|
|
|
|
fsm_reset_all_situations_transparencies_at_value (gtk_adjustment_get_value (adj_situ));
|
|
|
|
}
|
|
|
|
|
2024-07-26 22:18:03 +02:00
|
|
|
|