/* -*- Mode: C ; c-basic-offset: 2 -*- */ /* * LADI Session Handler (ladish) * * Copyright (C) 2009,2010,2011,2012 Nedko Arnaudov * ************************************************************************** * This file contains implementation of graph canvas object ************************************************************************** * * LADI Session Handler 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 2 of the License, or * (at your option) any later version. * * LADI Session Handler 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 LADI Session Handler. If not, see * or write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include "graph_canvas.h" #include "../dbus_constants.h" #include "../common/catdup.h" #include "internal.h" struct graph_canvas { graph_proxy_handle graph; canvas_handle canvas; void (* fill_menu)(GtkMenu * menu); struct list_head clients; }; struct client { struct list_head siblings; uint64_t id; canvas_module_handle canvas_module; struct list_head ports; struct graph_canvas * owner_ptr; unsigned int inport_count; unsigned int outport_count; }; struct port { struct list_head siblings; uint64_t id; bool is_input; canvas_port_handle canvas_port; struct graph_canvas * graph_canvas; }; static struct client * find_client( struct graph_canvas * graph_canvas_ptr, uint64_t id) { struct list_head * node_ptr; struct client * client_ptr; list_for_each(node_ptr, &graph_canvas_ptr->clients) { client_ptr = list_entry(node_ptr, struct client, siblings); if (client_ptr->id == id) { return client_ptr; } } return NULL; } static struct port * find_port( struct client * client_ptr, uint64_t id) { struct list_head * node_ptr; struct port * port_ptr; list_for_each(node_ptr, &client_ptr->ports) { port_ptr = list_entry(node_ptr, struct port, siblings); if (port_ptr->id == id) { return port_ptr; } } return NULL; } #define port1_ptr ((struct port *)port1_context) #define port2_ptr ((struct port *)port2_context) static void connect_request( void * port1_context, void * port2_context) { log_info("connect request(%"PRIu64", %"PRIu64")", port1_ptr->id, port2_ptr->id); ASSERT(port1_ptr->graph_canvas == port2_ptr->graph_canvas); graph_proxy_connect_ports(port1_ptr->graph_canvas->graph, port1_ptr->id, port2_ptr->id); } void disconnect_request( void * port1_context, void * port2_context) { log_info("disconnect request(%"PRIu64", %"PRIu64")", port1_ptr->id, port2_ptr->id); ASSERT(port1_ptr->graph_canvas == port2_ptr->graph_canvas); graph_proxy_disconnect_ports(port1_ptr->graph_canvas->graph, port1_ptr->id, port2_ptr->id); } #undef port1_ptr #undef port2_ptr #define client_ptr ((struct client *)module_context) void module_location_changed( void * module_context, double x, double y) { char x_str[100]; char y_str[100]; char * locale; log_info("module_location_changed(id = %3llu, x = %6.1f, y = %6.1f)", (unsigned long long)client_ptr->id, x, y); locale = strdup(setlocale(LC_NUMERIC, NULL)); if (locale == NULL) { log_error("strdup() failed for locale string"); return; } setlocale(LC_NUMERIC, "POSIX"); sprintf(x_str, "%f", x); sprintf(y_str, "%f", y); setlocale(LC_NUMERIC, locale); free(locale); graph_proxy_dict_entry_set( client_ptr->owner_ptr->graph, GRAPH_DICT_OBJECT_TYPE_CLIENT, client_ptr->id, URI_CANVAS_X, x_str); graph_proxy_dict_entry_set( client_ptr->owner_ptr->graph, GRAPH_DICT_OBJECT_TYPE_CLIENT, client_ptr->id, URI_CANVAS_Y, y_str); } static void on_popup_menu_action_client_rename(GtkWidget * UNUSED(menuitem), gpointer module_context) { log_info("on_popup_menu_action_client_rename %"PRIu64, client_ptr->id); char * new_name; if (name_dialog(_("Rename client"), _("Client name"), canvas_get_module_name(client_ptr->canvas_module), &new_name)) { if (!graph_proxy_rename_client(client_ptr->owner_ptr->graph, client_ptr->id, new_name)) { error_message_box("Rename failed"); } free(new_name); } } static void on_popup_menu_action_split(GtkWidget * UNUSED(menuitem), gpointer module_context) { //log_info("on_popup_menu_action_split"); graph_proxy_split(client_ptr->owner_ptr->graph, client_ptr->id); } static void on_popup_menu_action_remove(GtkWidget * UNUSED(menuitem), gpointer module_context) { //log_info("on_popup_menu_action_split"); if (!graph_proxy_remove_client(client_ptr->owner_ptr->graph, client_ptr->id)) { error_message_box("Client remove failed"); } } static void fill_module_menu(GtkMenu * menu, void * module_context) { GtkWidget * menuitem; log_info("fill_module_menu %"PRIu64, client_ptr->id); menuitem = gtk_menu_item_new_with_label(_("Client rename")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_client_rename, client_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); if (client_ptr->inport_count == 0 && client_ptr->outport_count == 0) { menuitem = gtk_menu_item_new_with_label(_("Remove")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_remove, client_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); } if (client_ptr->inport_count != 0 && client_ptr->outport_count != 0) { menuitem = gtk_menu_item_new_with_label(_("Split")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_split, client_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); } } #undef client_ptr #define port_ptr ((struct port *)port_context) static void on_popup_menu_action_port_rename(GtkWidget * UNUSED(menuitem), gpointer port_context) { log_info("on_popup_menu_action_port_rename %"PRIu64, port_ptr->id); char * new_name; if (name_dialog(_("Rename port"), _("Port name"), canvas_get_port_name(port_ptr->canvas_port), &new_name)) { if (!graph_proxy_rename_port(port_ptr->graph_canvas->graph, port_ptr->id, new_name)) { error_message_box("Rename failed"); } free(new_name); } } static void on_popup_menu_action_port_move(GtkWidget * UNUSED(menuitem), gpointer port_context) { struct client * client_ptr; log_info("on_popup_menu_action_port_move %"PRIu64, port_ptr->id); if (!canvas_get_one_selected_module(port_ptr->graph_canvas->canvas, (void **)&client_ptr)) { return; } if (!graph_proxy_move_port(port_ptr->graph_canvas->graph, port_ptr->id, client_ptr->id)) { error_message_box("Port move failed"); } } static void fill_port_menu(GtkMenu * menu, void * port_context) { GtkWidget * menuitem; log_info("fill_port_menu %"PRIu64, port_ptr->id); menuitem = gtk_menu_item_new_with_label(_("Rename")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_port_rename, port_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); if (canvas_get_selected_modules_count(port_ptr->graph_canvas->canvas) == 1) { menuitem = gtk_menu_item_new_with_label(_("Move port")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_port_move, port_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); } } #undef port_ptr #define canvas_ptr ((struct graph_canvas *)canvas_context) static void on_popup_menu_action_join_clients(GtkWidget * UNUSED(menuitem), gpointer canvas_context) { struct client * client1_ptr; struct client * client2_ptr; if (!canvas_get_two_selected_modules(canvas_ptr->canvas, (void **)&client1_ptr, (void **)&client2_ptr)) { return; } if (!graph_proxy_join(canvas_ptr->graph, client1_ptr->id, client2_ptr->id)) { error_message_box("Join failed"); } } static void on_popup_menu_action_new_client(GtkWidget * UNUSED(menuitem), gpointer canvas_context) { char * new_name; uint64_t client_id; if (name_dialog(_("New client"), _("Client name"), "", &new_name)) { if (!graph_proxy_new_client(canvas_ptr->graph, new_name, &client_id)) { error_message_box("New client creation failed"); } free(new_name); } } static void fill_canvas_menu(GtkMenu * menu, void * canvas_context) { GtkWidget * menuitem; log_info("fill_canvas_menu"); menuitem = gtk_menu_item_new_with_label(_("New client")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_new_client, canvas_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); if (canvas_get_selected_modules_count(canvas_ptr->canvas) == 2) { menuitem = gtk_menu_item_new_with_label(_("Join clients")); g_signal_connect(menuitem, "activate", (GCallback)on_popup_menu_action_join_clients, canvas_ptr); gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem); } canvas_ptr->fill_menu(menu); } #undef canvas_ptr bool graph_canvas_create( int width, int height, void (* fill_canvas_menu_callback)(GtkMenu * menu), graph_canvas_handle * graph_canvas_handle_ptr) { struct graph_canvas * graph_canvas_ptr; graph_canvas_ptr = malloc(sizeof(struct graph_canvas)); if (graph_canvas_ptr == NULL) { return false; } graph_canvas_ptr->fill_menu = fill_canvas_menu_callback; if (!canvas_create( width, height, graph_canvas_ptr, connect_request, disconnect_request, module_location_changed, fill_canvas_menu, fill_module_menu, fill_port_menu, &graph_canvas_ptr->canvas)) { free(graph_canvas_ptr); return false; } graph_canvas_ptr->graph = NULL; INIT_LIST_HEAD(&graph_canvas_ptr->clients); *graph_canvas_handle_ptr = (graph_canvas_handle)graph_canvas_ptr; return true; } #define graph_canvas_ptr ((struct graph_canvas *)graph_canvas) static void clear( void * graph_canvas) { log_info("canvas::clear()"); canvas_clear(graph_canvas_ptr->canvas); } static void client_appeared( void * graph_canvas, uint64_t id, const char * name) { struct client * client_ptr; char * x_str; char * y_str; double x; double y; char * locale; double width; double height; log_info("canvas::client_appeared(%"PRIu64", \"%s\")", id, name); canvas_get_size(graph_canvas_ptr->canvas, &width, &height); //log_debug("width %f, height %f", width, height); client_ptr = malloc(sizeof(struct client)); if (client_ptr == NULL) { log_error("allocation of memory for struct client failed"); return; } client_ptr->id = id; client_ptr->inport_count = 0; client_ptr->outport_count = 0; INIT_LIST_HEAD(&client_ptr->ports); client_ptr->owner_ptr = graph_canvas_ptr; x = 0; y = 0; if (!graph_proxy_dict_entry_get( client_ptr->owner_ptr->graph, GRAPH_DICT_OBJECT_TYPE_CLIENT, id, URI_CANVAS_X, &x_str)) { x_str = NULL; x = width / 2 - 200 + rand() % 300; } if (!graph_proxy_dict_entry_get( client_ptr->owner_ptr->graph, GRAPH_DICT_OBJECT_TYPE_CLIENT, id, URI_CANVAS_Y, &y_str)) { y_str = NULL; y = height / 2 - 200 + rand() % 300; } if (x_str != NULL || y_str != NULL) { locale = strdup(setlocale(LC_NUMERIC, NULL)); if (locale == NULL) { log_error("strdup() failed for locale string"); } else { setlocale(LC_NUMERIC, "POSIX"); if (x_str != NULL) { //log_info("converting %s", x_str); sscanf(x_str, "%lf", &x); } if (y_str != NULL) { //log_info("converting %s", y_str); sscanf(y_str, "%lf", &y); } setlocale(LC_NUMERIC, locale); free(locale); } if (x_str != NULL) { free(x_str); } if (y_str != NULL) { free(y_str); } } if (!canvas_create_module(graph_canvas_ptr->canvas, name, x, y, true, true, client_ptr, &client_ptr->canvas_module)) { log_error("canvas_create_module(\"%s\") failed", name); free(client_ptr); return; } if (x_str == NULL || y_str == NULL) { /* we have generated random value, store it */ module_location_changed(client_ptr, x, y); } list_add_tail(&client_ptr->siblings, &graph_canvas_ptr->clients); } static void client_disappeared( void * graph_canvas, uint64_t id) { struct client * client_ptr; log_info("canvas::client_disappeared(%"PRIu64")", id); client_ptr = find_client(graph_canvas_ptr, id); if (client_ptr == NULL) { log_error("cannot find disappearing client %"PRIu64"", id); return; } list_del(&client_ptr->siblings); canvas_destroy_module(graph_canvas_ptr->canvas, client_ptr->canvas_module); free(client_ptr); } static void client_renamed(void * graph_canvas, uint64_t id, const char * old_name, const char * new_name) { struct client * client_ptr; log_info("canvas::client_renamed(%"PRIu64", '%s', '%s')", id, old_name, new_name); client_ptr = find_client(graph_canvas_ptr, id); if (client_ptr == NULL) { log_error("cannot find renamed client %"PRIu64, id); return; } canvas_set_module_name(client_ptr->canvas_module, new_name); } static void port_appeared( void * graph_canvas, uint64_t client_id, uint64_t port_id, const char * port_name, bool is_input, bool UNUSED(is_terminal), bool is_midi) { int color; struct client * client_ptr; struct port * port_ptr; char * value; char * name_override; log_info("canvas::port_appeared(%"PRIu64", %"PRIu64", \"%s\")", client_id, port_id, port_name); client_ptr = find_client(graph_canvas_ptr, client_id); if (client_ptr == NULL) { log_error("cannot find client %"PRIu64" of appearing port %"PRIu64" \"%s\"", client_id, port_id, port_name); return; } port_ptr = malloc(sizeof(struct port)); if (port_ptr == NULL) { log_error("allocation of memory for struct port failed"); return; } port_ptr->id = port_id; port_ptr->is_input = is_input; port_ptr->graph_canvas = graph_canvas_ptr; // Darkest tango palette colour, with S -= 6, V -= 6, w/ transparency if (is_midi) { color = 0x960909C0; } else { color = 0x244678C0; } if (!graph_proxy_dict_entry_get( client_ptr->owner_ptr->graph, GRAPH_DICT_OBJECT_TYPE_PORT, port_id, URI_A2J_PORT, &value)) { name_override = NULL; } else { if (strcmp(value, "yes") == 0) { name_override = catdup(port_name, " (a2j)"); if (name_override != NULL) { port_name = name_override; } } else { name_override = NULL; } free(value); } if (!canvas_create_port(graph_canvas_ptr->canvas, client_ptr->canvas_module, port_name, is_input, color, port_ptr, &port_ptr->canvas_port)) { log_error("canvas_create_port(\"%s\") failed", port_name); free(port_ptr); free(name_override); return; } list_add_tail(&port_ptr->siblings, &client_ptr->ports); free(name_override); if (is_input) { client_ptr->inport_count++; } else { client_ptr->outport_count++; } } static void port_disappeared( void * graph_canvas, uint64_t client_id, uint64_t port_id) { struct client * client_ptr; struct port * port_ptr; log_info("canvas::port_disappeared(%"PRIu64", %"PRIu64")", client_id, port_id); client_ptr = find_client(graph_canvas_ptr, client_id); if (client_ptr == NULL) { log_error("cannot find client %"PRIu64" of disappearing port %"PRIu64"", client_id, port_id); return; } port_ptr = find_port(client_ptr, port_id); if (client_ptr == NULL) { log_error("cannot find disappearing port %"PRIu64" of client %"PRIu64"", port_id, client_id); return; } list_del(&port_ptr->siblings); canvas_destroy_port(graph_canvas_ptr->canvas, port_ptr->canvas_port); if (port_ptr->is_input) { client_ptr->inport_count--; } else { client_ptr->outport_count--; } free(port_ptr); } static void port_renamed( void * graph_canvas, uint64_t client_id, uint64_t port_id, const char * old_port_name, const char * new_port_name) { struct client * client_ptr; struct port * port_ptr; log_info("canvas::port_renamed(%"PRIu64", %"PRIu64", '%s', '%s')", client_id, port_id, old_port_name, new_port_name); client_ptr = find_client(graph_canvas_ptr, client_id); if (client_ptr == NULL) { log_error("cannot find client %"PRIu64" of renamed port %"PRIu64"", client_id, port_id); return; } port_ptr = find_port(client_ptr, port_id); if (port_ptr == NULL) { log_error("cannot find renamed port %"PRIu64" of client %"PRIu64"", port_id, client_id); return; } canvas_set_port_name(port_ptr->canvas_port, new_port_name); } static void ports_connected( void * graph_canvas, uint64_t client1_id, uint64_t port1_id, uint64_t client2_id, uint64_t port2_id) { struct client * client1_ptr; struct port * port1_ptr; struct client * client2_ptr; struct port * port2_ptr; log_info("canvas::ports_connected(%"PRIu64", %"PRIu64", %"PRIu64", %"PRIu64")", client1_id, port1_id, client2_id, port2_id); client1_ptr = find_client(graph_canvas_ptr, client1_id); if (client1_ptr == NULL) { log_error("cannot find client %"PRIu64" of connected port %"PRIu64"", client1_id, port1_id); return; } port1_ptr = find_port(client1_ptr, port1_id); if (port1_ptr == NULL) { log_error("cannot find connected port %"PRIu64" of client %"PRIu64"", port1_id, client1_id); return; } client2_ptr = find_client(graph_canvas_ptr, client2_id); if (client2_ptr == NULL) { log_error("cannot find client %"PRIu64" of connected port %"PRIu64"", client2_id, port2_id); return; } port2_ptr = find_port(client2_ptr, port2_id); if (port2_ptr == NULL) { log_error("cannot find connected port %"PRIu64" of client %"PRIu64"", port2_id, client2_id); return; } canvas_add_connection( graph_canvas_ptr->canvas, port1_ptr->canvas_port, port2_ptr->canvas_port, canvas_get_port_color(port1_ptr->canvas_port) + 0x22222200); } static void ports_disconnected( void * graph_canvas, uint64_t client1_id, uint64_t port1_id, uint64_t client2_id, uint64_t port2_id) { struct client * client1_ptr; struct port * port1_ptr; struct client * client2_ptr; struct port * port2_ptr; log_info("canvas::ports_disconnected(%"PRIu64", %"PRIu64", %"PRIu64", %"PRIu64")", client1_id, port1_id, client2_id, port2_id); client1_ptr = find_client(graph_canvas_ptr, client1_id); if (client1_ptr == NULL) { log_error("cannot find client %"PRIu64" of disconnected port %"PRIu64"", client1_id, port1_id); return; } port1_ptr = find_port(client1_ptr, port1_id); if (port1_ptr == NULL) { log_error("cannot find disconnected port %"PRIu64" of client %"PRIu64"", port1_id, client1_id); return; } client2_ptr = find_client(graph_canvas_ptr, client2_id); if (client2_ptr == NULL) { log_error("cannot find client %"PRIu64" of disconnected port %"PRIu64"", client2_id, port2_id); return; } port2_ptr = find_port(client2_ptr, port2_id); if (port2_ptr == NULL) { log_error("cannot find disconnected port %"PRIu64" of client %"PRIu64"", port2_id, client2_id); return; } canvas_remove_connection( graph_canvas_ptr->canvas, port1_ptr->canvas_port, port2_ptr->canvas_port); } void graph_canvas_destroy( graph_canvas_handle graph_canvas) { if (graph_canvas_ptr->graph != NULL) { graph_canvas_detach(graph_canvas); } free(graph_canvas_ptr); } bool graph_canvas_attach( graph_canvas_handle graph_canvas, graph_proxy_handle graph) { ASSERT(graph_canvas_ptr->graph == NULL); if (!graph_proxy_attach( graph, graph_canvas, clear, client_appeared, client_renamed, client_disappeared, port_appeared, port_renamed, port_disappeared, ports_connected, ports_disconnected)) { return false; } graph_canvas_ptr->graph = graph; return true; } void graph_canvas_detach( graph_canvas_handle graph_canvas) { ASSERT(graph_canvas_ptr->graph != NULL); graph_proxy_detach(graph_canvas_ptr->graph, graph_canvas); graph_canvas_ptr->graph = NULL; } canvas_handle graph_canvas_get_canvas( graph_canvas_handle graph_canvas) { return graph_canvas_ptr->canvas; }