bugfixes and clear button in mappings table
[drnoksnes] / gui / plugin.c
index 725dda4..71ac408 100644 (file)
 #include <gconf/gconf.h>
 #include <gconf/gconf-client.h>
 #include <hildon/hildon-file-chooser-dialog.h>
+#include <hildon/hildon-note.h>
 
 #include "../platform/hgw.h"
+#include "plugin.h"
 
 static GtkWidget * load_plugin(void);
 static void unload_plugin(void);
@@ -39,9 +41,9 @@ static GtkWidget ** load_menu(guint *);
 static void update_menu(void);
 static void plugin_callback(GtkWidget * menu_item, gpointer data);
 
-GConfClient *gcc = NULL;
+GConfClient * gcc = NULL;
 static GameStartupInfo gs;
-GtkWidget *menu_items[2];
+static GtkWidget * menu_items[1];
 
 static StartupPluginInfo plugin_info = {
        load_plugin,
@@ -52,37 +54,49 @@ static StartupPluginInfo plugin_info = {
        plugin_callback
 };
 
-static const gchar * rom_globs[] = {
-       "*.smc",
-       "*.fig",
-       "*.sfc",
-       NULL
-};
-
 STARTUP_INIT_PLUGIN(plugin_info, gs, FALSE, TRUE)
 
-// Yes, I'm using the label not only to show but also save the current value.
-static GtkLabel * rom_label;
+gchar* current_rom_file = 0;
+static GtkLabel* rom_label;
+static GtkCheckButton* audio_check;
+static GtkCheckButton* turbo_check;
+static GtkSpinButton* frameskip_spin;
+static GtkCheckButton* auto_framerate_check;
+static GtkCheckButton* trans_check;
+static GtkComboBox* speedhacks_combo;
 
-static gchar *
-interface_file_chooser
-(GtkWindow * parent, GtkFileChooserAction action, const gchar ** extension)
+static void set_rom(const char * rom_file)
+{
+       if (!rom_file) return;
+       if (current_rom_file) g_free(current_rom_file);
+
+       current_rom_file = g_strdup(rom_file);
+       gtk_label_set_text(GTK_LABEL(rom_label), rom_file); // TODO UTF-8
+
+       game_state_update();
+       save_clear();
+}
+
+static inline GtkWindow* get_parent_window() {
+       return GTK_WINDOW(gs.ui->hildon_appview);
+}
+
+static void select_rom_callback(GtkWidget * button, gpointer data)
 {
        GtkWidget * dialog;
        GtkFileFilter * filter;
-       const gchar * current_filename;
+       const gchar * current_filename = gtk_label_get_text(rom_label);
        gchar * filename = NULL;
-       int i;
 
        filter = gtk_file_filter_new();
-       for (i = 0; extension[i]; i++) {
-               gtk_file_filter_add_pattern(filter, extension[i]);
-       }
+       gtk_file_filter_add_pattern(filter, "*.smc");
+       gtk_file_filter_add_pattern(filter, "*.sfc");
+       gtk_file_filter_add_pattern(filter, "*.fig");
 
-       dialog = hildon_file_chooser_dialog_new_with_properties(parent, 
-               "action", action, "local_only", TRUE, "filter", filter, NULL);
+       dialog = hildon_file_chooser_dialog_new_with_properties(
+               get_parent_window(),
+               "action", GTK_FILE_CHOOSER_ACTION_OPEN, "filter", filter, NULL);
 
-       current_filename = gtk_label_get_text(rom_label);
        if (current_filename && strlen(current_filename) > 1) {
                // By default open showing the last selected file
                gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), 
@@ -96,21 +110,25 @@ interface_file_chooser
 
        gtk_widget_destroy(dialog);
 
-       return filename;
+       if (filename) {
+               set_rom(filename);
+               g_free(filename);
+       }
 }
 
-static void select_rom_callback(GtkWidget * button, gpointer data)
+static void controls_item_callback(GtkWidget * button, gpointer data)
 {
-       gchar * filename = interface_file_chooser(
-               GTK_WINDOW(gtk_widget_get_parent_window(button)),
-               GTK_FILE_CHOOSER_ACTION_OPEN,
-               rom_globs);
+       controls_setup();
+       controls_dialog(get_parent_window());
+}
 
-       if (!filename) return;
-       
-       gtk_label_set_text(rom_label, filename);
-       
-       g_free(filename);
+static void auto_framerate_callback(GtkWidget * button, gpointer data)
+{
+       if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(auto_framerate_check))) {
+               gtk_widget_set_sensitive(GTK_WIDGET(frameskip_spin), FALSE);
+       } else {
+               gtk_widget_set_sensitive(GTK_WIDGET(frameskip_spin), TRUE);
+       }
 }
 
 static GtkWidget * load_plugin(void)
@@ -119,52 +137,166 @@ static GtkWidget * load_plugin(void)
        gcc = gconf_client_get_default();
 
        GtkWidget* parent = gtk_vbox_new(FALSE, HILDON_MARGIN_DEFAULT);
-       GtkWidget* parent_hbox = gtk_hbox_new(FALSE, HILDON_MARGIN_DEFAULT);
+       GtkWidget* rom_hbox = gtk_hbox_new(FALSE, HILDON_MARGIN_DEFAULT);
+       GtkWidget* opt_hbox = gtk_hbox_new(FALSE, HILDON_MARGIN_DEFAULT);
+       GtkWidget* opt2_hbox = gtk_hbox_new(FALSE, HILDON_MARGIN_DEFAULT);
+
        GtkWidget* selectRomBtn = gtk_button_new_with_label("Select ROM...");
        rom_label = GTK_LABEL(gtk_label_new(NULL));
-       
+
+       GtkContainer* audio_cont =
+               GTK_CONTAINER(gtk_alignment_new(0.0, 0.0, 0.0, 0.0));
+       audio_check =
+               GTK_CHECK_BUTTON(gtk_check_button_new_with_label("Enable audio"));
+       GtkWidget* framerate_label = gtk_label_new("Frameskip:");
+       frameskip_spin =
+               GTK_SPIN_BUTTON(gtk_spin_button_new_with_range(0.0, 10.0, 1.0));
+       auto_framerate_check =
+               GTK_CHECK_BUTTON(gtk_check_button_new_with_label("Auto"));
+       turbo_check =
+               GTK_CHECK_BUTTON(gtk_check_button_new_with_label("Turbo mode"));
+       GtkContainer* trans_cont =
+               GTK_CONTAINER(gtk_alignment_new(0.0, 0.0, 0.0, 0.0));
+       trans_check =
+               GTK_CHECK_BUTTON(gtk_check_button_new_with_label("Accurate graphics"));
+       speedhacks_combo =
+               GTK_COMBO_BOX(gtk_combo_box_new_text());
+
        gtk_widget_set_size_request(GTK_WIDGET(selectRomBtn),
-                                                               180, 50);
-                                                               
-       gtk_label_set_text(rom_label,
-                                               gconf_client_get_string(gcc, kGConfRomFile, NULL));
-                                                               
+                                                               180, 46);
+       gtk_combo_box_append_text(speedhacks_combo, "No speedhacks");
+       gtk_combo_box_append_text(speedhacks_combo, "Safe hacks only");
+       gtk_combo_box_append_text(speedhacks_combo, "All speedhacks");
 
-       g_signal_connect(G_OBJECT(selectRomBtn), "clicked",
-                                       G_CALLBACK (select_rom_callback), NULL);
 
-       gtk_box_pack_start(GTK_BOX(parent_hbox), selectRomBtn, FALSE, FALSE, 0);
-       gtk_box_pack_end(GTK_BOX(parent_hbox), GTK_WIDGET(rom_label), TRUE, TRUE, 0);
-       gtk_box_pack_start(GTK_BOX(parent), parent_hbox, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(rom_hbox), selectRomBtn, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(rom_hbox), GTK_WIDGET(rom_label), TRUE, TRUE, 0);
+
+       gtk_container_add(audio_cont, GTK_WIDGET(audio_check));
+       gtk_box_pack_start(GTK_BOX(opt_hbox), GTK_WIDGET(audio_cont), TRUE, TRUE, 0);
+       gtk_box_pack_start(GTK_BOX(opt_hbox), framerate_label, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(opt_hbox), GTK_WIDGET(frameskip_spin), FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(opt_hbox), GTK_WIDGET(auto_framerate_check), FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(opt_hbox), GTK_WIDGET(turbo_check), FALSE, FALSE, 0);
+
+       gtk_container_add(trans_cont, GTK_WIDGET(trans_check));
+       gtk_box_pack_start(GTK_BOX(opt2_hbox), GTK_WIDGET(trans_cont), TRUE, TRUE, 0);
+       gtk_box_pack_start(GTK_BOX(opt2_hbox), GTK_WIDGET(speedhacks_combo), FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(parent), rom_hbox, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(parent), opt_hbox, FALSE, FALSE, 0);
+       gtk_box_pack_start(GTK_BOX(parent), opt2_hbox, FALSE, FALSE, 0);
+
+       // Load current configuration from GConf
+       gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(audio_check),
+               !gconf_client_get_bool(gcc, kGConfDisableAudio, NULL));
+       gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(turbo_check),
+               gconf_client_get_bool(gcc, kGConfTurboMode, NULL));
+
+       int frameskip = gconf_client_get_int(gcc, kGConfFrameskip, NULL);
+       if (frameskip <= 0) {
+               gtk_widget_set_sensitive(GTK_WIDGET(frameskip_spin), FALSE);
+               gtk_spin_button_set_value(frameskip_spin, 0);
+               gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(auto_framerate_check), TRUE);
+       } else {
+               gtk_widget_set_sensitive(GTK_WIDGET(frameskip_spin), TRUE);
+               gtk_spin_button_set_value(frameskip_spin, frameskip - 1);
+               gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(auto_framerate_check), FALSE);
+       }
+
+       gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(trans_check),
+               gconf_client_get_bool(gcc, kGConfTransparency, NULL));
+
+       gtk_combo_box_set_active(speedhacks_combo,
+               gconf_client_get_int(gcc, kGConfSpeedhacks, NULL));
+
+       set_rom(gconf_client_get_string(gcc, kGConfRomFile, NULL));
+
+       // Connect signals
+       g_signal_connect(G_OBJECT(selectRomBtn), "clicked",
+                                       G_CALLBACK(select_rom_callback), NULL);
+       g_signal_connect(G_OBJECT(auto_framerate_check), "toggled",
+                                       G_CALLBACK(auto_framerate_callback), NULL);
 
        return parent;
 }
 
 static void unload_plugin(void)
 {
+       if (current_rom_file) {
+               g_free(current_rom_file);
+               current_rom_file = 0;
+       }
+       game_state_clear();
+       save_clear();
        g_object_unref(gcc);
 }
 
 static void write_config(void)
 {
-       gconf_client_set_string(gcc, kGConfRomFile,
-        gtk_label_get_text(GTK_LABEL(rom_label)), NULL);
+       // Write settings to GConf
+       gconf_client_set_bool(gcc, kGConfDisableAudio,
+               !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(audio_check)), NULL);
+       gconf_client_set_bool(gcc, kGConfTurboMode,
+               gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(turbo_check)), NULL);
+       if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(auto_framerate_check))) {
+               gconf_client_set_int(gcc, kGConfFrameskip, 0, NULL);
+       } else {
+               gconf_client_set_int(gcc, kGConfFrameskip,
+                       gtk_spin_button_get_value(frameskip_spin) + 1, NULL);
+       }
+       gconf_client_set_bool(gcc, kGConfTransparency,
+               gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(trans_check)), NULL);
+       gconf_client_set_int(gcc, kGConfSpeedhacks,
+               gtk_combo_box_get_active(speedhacks_combo), NULL);
+
+       if (current_rom_file) {
+               gconf_client_set_string(gcc, kGConfRomFile, current_rom_file, NULL);
+       } else {
+               GtkWidget* note = hildon_note_new_information(get_parent_window(),
+                       "No ROM selected");
+               gtk_dialog_run(GTK_DIALOG(note));
+               gtk_widget_destroy(note);
+       }
+
+       controls_setup();
 }
 
 static GtkWidget **load_menu(guint *nitems)
 {
-       *nitems = 0;
+       menu_items[0] = gtk_menu_item_new_with_label("Settings");
+       *nitems = 1;
+
+       GtkMenu* settings_menu = GTK_MENU(gtk_menu_new());
+       GtkMenuItem* controls_item =
+               GTK_MENU_ITEM(gtk_menu_item_new_with_label("Controls…"));
        
-       return NULL;
+       gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_items[0]),
+               GTK_WIDGET(settings_menu));
+       gtk_menu_append(GTK_MENU(settings_menu), GTK_WIDGET(controls_item));
+
+       g_signal_connect(G_OBJECT(controls_item), "activate",
+                                       G_CALLBACK(controls_item_callback), NULL);
+
+       return menu_items;
 }
 
 static void update_menu(void)
 {
-
+       // Nothing to update in the current menu
 }
 
 static void plugin_callback(GtkWidget * menu_item, gpointer data)
 {
-
+       switch ((gint) data) {
+               case 20:        // ME_GAME_OPEN
+                       save_load(get_parent_window());
+                       break;
+               case 21:        // ME_GAME_SAVE
+                       save_save(get_parent_window());
+                       break;
+               case 22:        // ME_GAME_SAVE_AS
+                       save_save_as(get_parent_window());
+                       break;
+       }
 }