/* * Copyright (c) 2009 Tomasz Moń * * This program 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; under version 3 of the License. * * 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include "gdigi.h" #include "gui.h" #include "effects.h" #include "preset.h" #include "gtkknob.h" #include "knob.h" extern EffectList effects[]; extern int n_effects; typedef struct { GtkObject *widget; gint id; gint position; /* used for combo boxes, if widget isn't combo box, then both value and x are -1 */ gint value; /* effect type value */ gint x; /* combo box item number */ } WidgetListElem; static GtkKnobAnim *knob_anim = NULL; /* animation used by knobs */ static GList *widget_list = NULL; /* this list contains WidgetListElem data elements */ static gboolean allow_send = FALSE; /* if FALSE GUI parameter changes won't be sent to device */ /** * show_error_message: * @parent: transient parent, or NULL for none * @message: error description * * Shows error message dialog. **/ void show_error_message(GtkWidget *parent, gchar *message) { g_return_if_fail(message != NULL); GtkWidget *msg = gtk_message_dialog_new(GTK_WINDOW(parent), GTK_DIALOG_MODAL, GTK_MESSAGE_ERROR, GTK_BUTTONS_OK, message); gtk_dialog_run(GTK_DIALOG(msg)); gtk_widget_destroy(msg); } /** * value_changed_option_cb: * @adj: the object which emitted the signal * @setting: setting controlled by adj * * Sets effect value. **/ void value_changed_option_cb(GtkAdjustment *adj, EffectSettings *setting) { g_return_if_fail(setting != NULL); if (allow_send) { gdouble val; g_object_get(G_OBJECT(adj), "value", &val, NULL); set_option(setting->id, setting->position, (gint)val); } if (setting->values != NULL && setting->values->labels != NULL) { GtkWidget *label; gint x; gdouble val = -1.0; g_object_get(G_OBJECT(adj), "value", &val, NULL); x = (gint)val; if ((x >= setting->values->min) && (x <= setting->values->max)) { label = g_object_get_data(G_OBJECT(adj), "label"); gtk_label_set_text(GTK_LABEL(label), setting->values->labels[x]); } } } /** * toggled_cb: * @button: the object which emitted the signal * @effect: effect controlled by button * * Turns effect on/off basing on state of button. **/ void toggled_cb(GtkToggleButton *button, Effect *effect) { g_return_if_fail(effect != NULL); if (allow_send) { guint val = gtk_toggle_button_get_active(button); set_option(effect->id, effect->position, val); } } /** * widget_list_add: * @widget: GtkObject to add to widget list * @id: object controlled ID * @position: object controlled position * @value: effect value type (if widget is GtkComboBox, otherwise -1) * @x: combo box item number (if widget is GtkComboBox, otherwise -1) * * Adds widget to widget list. **/ static void widget_list_add(GtkObject *widget, gint id, gint position, gint value, gint x) { WidgetListElem *el; el = g_slice_new(WidgetListElem); el->widget = widget; el->id = id; el->position = position; el->value = value; el->x = x; widget_list = g_list_prepend(widget_list, el); } /** * apply_widget_setting: * @el: widget list element * @param: parameter to set * * Sets widget list element value to param value. **/ static void apply_widget_setting(WidgetListElem *el, SettingParam *param) { if ((el->id == param->id) && (el->position == param->position)) { if (el->value == -1) { if (GTK_IS_TOGGLE_BUTTON(el->widget)) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(el->widget), (param->value == 0) ? FALSE : TRUE); else if (GTK_IS_ADJUSTMENT(el->widget)) gtk_adjustment_set_value(GTK_ADJUSTMENT(el->widget), (gdouble)param->value); } else { /* combo box */ if (el->value == param->value) gtk_combo_box_set_active(GTK_COMBO_BOX(el->widget), el->x); } } } /** * apply_preset_to_gui: * @preset: preset to sync * * Synces GUI with preset. **/ static void apply_preset_to_gui(Preset *preset) { g_return_if_fail(preset != NULL); g_return_if_fail(widget_list != NULL); allow_send = FALSE; GList *iter = preset->params; while (iter) { SettingParam *param = iter->data; iter = iter->next; if (param != NULL) g_list_foreach(widget_list, (GFunc)apply_widget_setting, param); } allow_send = TRUE; } /** * apply_current_preset: * * Synces GUI with device current edit buffer. **/ static void apply_current_preset() { GString *msg = get_current_preset(); Preset *preset = create_preset_from_data(msg); g_string_free(msg, TRUE); apply_preset_to_gui(preset); preset_free(preset); } /** * create_table: * @settings: effect parameters * @amt: amount of effect parameters * * Creates knobs that allow user to set effect parameters. * * Return value: GtkTable containing necessary widgets to set effect parameters. **/ GtkWidget *create_table(EffectSettings *settings, gint amt) { GtkWidget *table, *label, *widget, *knob; GtkObject *adj; int x; table = gtk_table_new(3, amt, FALSE); for (x = 0; xmin, settings[x].values->max, 1.0, /* step increment */ MAX((settings[x].values->max / 100), 5.0), /* page increment */ 0.0); knob = gtk_knob_new(GTK_ADJUSTMENT(adj), knob_anim); if (settings[x].values->labels == NULL) { widget = gtk_spin_button_new(GTK_ADJUSTMENT(adj), 1.0, 0); gtk_spin_button_set_numeric(GTK_SPIN_BUTTON(widget), TRUE); gtk_spin_button_set_update_policy(GTK_SPIN_BUTTON(widget), GTK_UPDATE_IF_VALID); } else { widget = gtk_label_new(settings[x].values->labels[0]); g_object_set_data(G_OBJECT(adj), "label", widget); } widget_list_add(adj, settings[x].id, settings[x].position, -1, -1); gtk_table_attach(GTK_TABLE(table), label, 0, 1, x, x+1, GTK_SHRINK, GTK_SHRINK, 2, 2); gtk_table_attach(GTK_TABLE(table), knob, 1, 2, x, x+1, GTK_SHRINK, GTK_SHRINK, 2, 2); gtk_table_attach(GTK_TABLE(table), widget, 2, 3, x, x+1, GTK_SHRINK, GTK_SHRINK, 2, 2); g_signal_connect(G_OBJECT(adj), "value-changed", G_CALLBACK(value_changed_option_cb), &settings[x]); } return table; } /** * create_on_off_button: * @effect: Effect that can be turned on/off * * Creates toggle button that allow user to turn effect on/off. * * Return value: GtkToggleButton **/ GtkWidget *create_on_off_button(Effect *effect) { GtkWidget *button = gtk_toggle_button_new_with_label(effect->label); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), FALSE); g_signal_connect(G_OBJECT(button), "toggled", G_CALLBACK(toggled_cb), effect); widget_list_add(GTK_OBJECT(button), effect->id, effect->position, -1, -1); return button; } typedef struct { gint type; /* effect group type (value) */ gint id; /* option ID */ gint position; /* position */ GtkWidget *child; /* child widget */ } EffectSettingsGroup; /** * effect_settings_group_free: * @group: group to be freed * * Frees all memory used by group **/ void effect_settings_group_free(EffectSettingsGroup *group) { /* destroy widget without parent */ if (gtk_widget_get_parent(group->child) == NULL) gtk_widget_destroy(group->child); g_object_unref(group->child); g_slice_free(EffectSettingsGroup, group); } /** * combo_box_changed_cb: * @widget: the object which emitted the signal * @data: user data (unused, can be anything) * * Switches effect type and shows widgets allowing to set selected effect type parameters. **/ void combo_box_changed_cb(GtkComboBox *widget, gpointer data) { GtkWidget *child; GtkWidget *vbox; EffectSettingsGroup *settings = NULL; gchar *name = NULL; gint x; g_object_get(G_OBJECT(widget), "active", &x, NULL); vbox = g_object_get_data(G_OBJECT(widget), "vbox"); if (x != -1) { name = g_strdup_printf("SettingsGroup%d", x); settings = g_object_get_data(G_OBJECT(widget), name); g_free(name); if (settings != NULL && allow_send) set_option(settings->id, settings->position, settings->type); child = g_object_get_data(G_OBJECT(widget), "active_child"); if (child != NULL) { gtk_container_remove(GTK_CONTAINER(gtk_widget_get_parent(gtk_widget_get_parent(vbox))), child); } gtk_container_add(GTK_CONTAINER(gtk_widget_get_parent(gtk_widget_get_parent(vbox))), settings->child); gtk_widget_show_all(gtk_widget_get_parent(gtk_widget_get_parent(vbox))); g_object_set_data(G_OBJECT(widget), "active_child", settings->child); } } /** * create_widget_container: * @group: Effect type groups * @amt: amount of effect groups * * Creates widget allowing user to choose effect type. * * Return value: widget that allow user to set effect type. **/ GtkWidget *create_widget_container(EffectGroup *group, gint amt) { GtkWidget *vbox; GtkWidget *widget; GtkWidget *combo_box = NULL; EffectSettingsGroup *settings = NULL; gchar *name = NULL; gint x; gint cmbox_no = -1; vbox = gtk_vbox_new(FALSE, 0); for (x = 0; xid = group[x].id; settings->type = group[x].type; settings->position = group[x].position; settings->child = widget; widget_list_add(GTK_OBJECT(combo_box), group[x].id, group[x].position, group[x].type, x); name = g_strdup_printf("SettingsGroup%d", cmbox_no); g_object_set_data_full(G_OBJECT(combo_box), name, settings, ((GDestroyNotify)effect_settings_group_free)); g_free(name); } else { widget = create_table(group[x].settings, group[x].settings_amt); gtk_container_add(GTK_CONTAINER(vbox), widget); } } return vbox; } /** * create_vbox: * @widgets: Effect descriptions * @amt: amount of effect descriptions * * Creates vbox containing widgets allowing user to set effect options. * * Return value: widget that allow user to set effect options. **/ GtkWidget *create_vbox(Effect *widgets, gint amt) { GtkWidget *vbox; GtkWidget *hbox; GtkWidget *widget; GtkWidget *table; int x; vbox = gtk_vbox_new(FALSE, 0); hbox = gtk_hbox_new(FALSE, 0); gtk_box_set_homogeneous(GTK_BOX(hbox), TRUE); for (x = 0; xvbox), table); cmbox = gtk_combo_box_new_text(); names = query_preset_names(PRESETS_USER); for (x=0; xvbox); if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) { gint number = gtk_combo_box_get_active(GTK_COMBO_BOX(cmbox)); if (number != -1) { store_preset_name(number, gtk_entry_get_text(GTK_ENTRY(entry))); } } gtk_widget_destroy(dialog); } /** * action_store_cb: * @action: the object which emitted the signal * * Shows store preset window. **/ static void action_store_cb(GtkAction *action) { GtkWidget *window = g_object_get_data(G_OBJECT(action), "window"); show_store_preset_window(window, NULL); } /** * action_show_about_dialog_cb: * @action: the object which emitted the signal * * Shows about dialog. **/ static void action_show_about_dialog_cb(GtkAction *action) { static const gchar * const authors[] = { "Tomasz Moń ", NULL }; static const gchar copyright[] = "Copyright \xc2\xa9 2009 Tomasz Moń"; static const gchar website[] = "http://desowin.org/gdigi/"; GtkWidget *window = g_object_get_data(G_OBJECT(action), "window"); gtk_show_about_dialog(GTK_WINDOW(window), "authors", authors, "copyright", copyright, "website", website, NULL); } typedef struct { gchar *name; gchar *suffix; } SupportedFileTypes; static SupportedFileTypes file_types[] = { {"RP250Preset", "*.rp250p"}, }; static guint n_file_types = G_N_ELEMENTS(file_types); /** * action_open_preset_cb: * @action: the object which emitted the signal * * Shows file chooser dialog. * If user opens valid preset file, the preset gets applied to edit buffer and store preset window is shown. **/ static void action_open_preset_cb(GtkAction *action) { static GtkWidget *dialog = NULL; if (dialog != NULL) return; GtkWidget *window = g_object_get_data(G_OBJECT(action), "window"); dialog = gtk_file_chooser_dialog_new("Open Preset", GTK_WINDOW(window), GTK_FILE_CHOOSER_ACTION_OPEN, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT, NULL); GtkFileFilter *filter; filter = gtk_file_filter_new(); gtk_file_filter_set_name(filter, "All Supported Types"); gtk_file_chooser_add_filter(GTK_FILE_CHOOSER(dialog), filter); int x; for (x=0; xparams; gint len = g_list_length(iter); g_string_append_printf(msg, "%c%c", ((len & 0xFF00) >> 8), (len & 0xFF)); while (iter) { SettingParam *param = iter->data; iter = iter->next; g_string_append_printf(msg, "%c%c%c", ((param->id & 0xFF00) >> 8), (param->id & 0xFF), param->position); append_value(msg, param->value); }; GString *start = g_string_new(NULL); g_string_append_printf(start, "%c%c%s%c%c%c", PRESETS_EDIT_BUFFER, 0, preset->name, 0 /* NULL terminated string */, 0 /* modified */, 2 /* messages to follow */); send_message(RECEIVE_PRESET_START, start->str, start->len); send_message(RECEIVE_PRESET_PARAMETERS, msg->str, msg->len); send_message(RECEIVE_PRESET_END, NULL, 0); show_store_preset_window(window, preset->name); g_string_free(start, TRUE); g_string_free(msg, TRUE); preset_free(preset); loaded = TRUE; } g_free(filename); } gtk_widget_destroy(dialog); dialog = NULL; } /** * widget_list_free: * @list: widget list to be freed * * Frees all memory used by widget list. */ static void widget_list_free(GList *list) { GList *iter; for (iter = list; iter; iter = iter->next) { g_slice_free(WidgetListElem, iter->data); } g_list_free(list); } /** * action_quit_cb: * @action: the object which emitted the signal * * Destroys action object "window" data, then stops gtk main loop. **/ static void action_quit_cb(GtkAction *action) { GtkWidget *window = g_object_get_data(G_OBJECT(action), "window"); gtk_widget_destroy(window); gtk_main_quit(); } static GtkActionEntry entries[] = { {"File", NULL, "_File"}, {"Preset", NULL, "_Preset"}, {"Help", NULL, "_Help"}, {"Open", GTK_STOCK_OPEN, "_Open", "O", "Open preset file", G_CALLBACK(action_open_preset_cb)}, {"Quit", GTK_STOCK_QUIT, "_Quit", "Q", "Quit", G_CALLBACK(action_quit_cb)}, {"Store", NULL, "_Store", "S", "Store", G_CALLBACK(action_store_cb)}, {"About", GTK_STOCK_ABOUT, "_About", "A", "About", G_CALLBACK(action_show_about_dialog_cb)}, }; static guint n_entries = G_N_ELEMENTS(entries); static const gchar *menu_info = "" " " " " " " " " " " " " " " " " " " " " " " " " " " ""; /** * add_action_data: * @ui: GtkUIManager to lookup actions * @path: path to action * @window: toplevel window * * Sets action object "window" data to toplevel window. **/ static void add_action_data(GtkUIManager *ui, const gchar *path, GtkWidget *window) { GtkAction *action; action = gtk_ui_manager_get_action(ui, path); g_return_if_fail(action != NULL); g_object_set_data(G_OBJECT(action), "window", window); } /** * add_menubar: * @window: toplevel window * @vbox: vbox to hold menubar * * Creates menubar (adds accel group to toplevel window as well) and packs it into vbox. **/ static void add_menubar(GtkWidget *window, GtkWidget *vbox) { GtkUIManager *ui; GtkActionGroup *actions; GError *error = NULL; actions = gtk_action_group_new("Actions"); gtk_action_group_add_actions(actions, entries, n_entries, NULL); ui = gtk_ui_manager_new(); gtk_ui_manager_insert_action_group(ui, actions, 0); g_object_unref(actions); gtk_window_add_accel_group(GTK_WINDOW(window), gtk_ui_manager_get_accel_group(ui)); if (!gtk_ui_manager_add_ui_from_string(ui, menu_info, -1, &error)) { g_message("building menus failed: %s", error->message); g_error_free(error); error = NULL; } gtk_box_pack_start(GTK_BOX(vbox), gtk_ui_manager_get_widget(ui, "/MenuBar"), FALSE, FALSE, 0); add_action_data(ui, "/MenuBar/File/Quit", window); add_action_data(ui, "/MenuBar/File/Open", window); add_action_data(ui, "/MenuBar/Preset/Store", window); add_action_data(ui, "/MenuBar/Help/About", window); g_object_unref(ui); } /** * gui_create: * * Creates main window. **/ void gui_create() { GtkWidget *window; GtkWidget *vbox; GtkWidget *hbox; GtkWidget *widget; GtkWidget *sw; /* scrolled window to carry preset treeview */ gint x; window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title(GTK_WINDOW(window), "gdigi"); vbox = gtk_vbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(window), vbox); add_menubar(window, vbox); hbox = gtk_hbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(vbox), hbox); sw = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); gtk_box_pack_start(GTK_BOX(hbox), sw, FALSE, FALSE, 0); widget = create_preset_tree(); gtk_container_add(GTK_CONTAINER(sw), widget); vbox = gtk_vbox_new(FALSE, 0); gtk_box_pack_start(GTK_BOX(hbox), vbox, TRUE, TRUE, 2); knob_anim = gtk_knob_animation_new_from_inline(knob_pixbuf); for (x = 0; x