2 * This file is part of DrNokSnes
4 * Copyright (C) 2009 Javier S. Pedro <maemo@javispedro.com>
6 * This software is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public License
8 * as published by the Free Software Foundation; either version 2.1 of
9 * the License, or (at your option) any later version.
11 * This software is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this software; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
24 #include <hildon/hildon-helper.h>
26 #include "../platform/hgw.h"
28 #include "cellrendererkey.h"
30 static GtkDialog* dialog;
31 static GtkComboBox* combo;
32 static GtkLabel* none_label;
33 static GtkScrolledWindow* keys_scroll;
34 static GtkListStore* keys_store;
35 static GtkTreeView* keys_list;
36 static GtkLabel* ts_label;
45 typedef struct ButtonEntry {
47 const char * gconf_key;
48 unsigned char scancode;
49 unsigned char default_scancode;
51 #define BUTTON_INITIALIZER(desc, name, default) \
52 { desc, kGConfKeysPath "/" name, 0, default }
55 static ButtonEntry buttons[] = {
56 BUTTON_INITIALIZER("A", "a", 48),
57 BUTTON_INITIALIZER("B", "b", 20),
58 BUTTON_INITIALIZER("X", "x", 32),
59 BUTTON_INITIALIZER("Y", "y", 45),
60 BUTTON_INITIALIZER("L", "l", 24),
61 BUTTON_INITIALIZER("R", "r", 22),
62 BUTTON_INITIALIZER("Start", "start", 65),
63 BUTTON_INITIALIZER("Select", "select", 135),
64 BUTTON_INITIALIZER("Up", "up", 111),
65 BUTTON_INITIALIZER("Down", "down", 116),
66 BUTTON_INITIALIZER("Left", "left", 113),
67 BUTTON_INITIALIZER("Right", "right", 114),
68 BUTTON_INITIALIZER("Return to launcher", "quit", 9),
69 BUTTON_INITIALIZER("Fullscreen", "fullscreen", 72),
73 static void show_widgets()
75 gtk_widget_show_all(GTK_WIDGET(combo));
76 gtk_widget_hide_all(GTK_WIDGET(none_label));
77 gtk_widget_hide_all(GTK_WIDGET(keys_scroll));
78 gtk_widget_hide_all(GTK_WIDGET(ts_label));
79 switch (gtk_combo_box_get_active(combo)) {
81 gtk_widget_show_all(GTK_WIDGET(none_label));
84 gtk_widget_show_all(GTK_WIDGET(keys_scroll));
87 gtk_widget_show_all(GTK_WIDGET(ts_label));
92 static gboolean load_key_config(GtkTreeModel *model, GtkTreePath *path,
93 GtkTreeIter *iter, gpointer data)
95 ButtonEntry *button_entry;
97 gtk_tree_model_get(model, iter,
98 BUTTONENTRY_COLUMN, &button_entry,
101 int scancode = gconf_client_get_int(gcc, button_entry->gconf_key, NULL);
102 button_entry->scancode = scancode;
104 gtk_tree_model_row_changed(GTK_TREE_MODEL(keys_store), path, iter);
109 static void load_config()
111 GConfValue* mapping = gconf_client_get(gcc, kGConfMapping, NULL);
114 mapping = gconf_value_new(GCONF_VALUE_INT);
115 gconf_value_set_int(mapping, 1);
116 gconf_client_set(gcc, kGConfMapping, mapping, NULL);
119 gtk_combo_box_set_active(combo, gconf_value_get_int(mapping));
121 gconf_client_preload(gcc, kGConfKeysPath, GCONF_CLIENT_PRELOAD_ONELEVEL, NULL);
122 gtk_tree_model_foreach(GTK_TREE_MODEL(keys_store), load_key_config, NULL);
125 gconf_value_free(mapping);
129 accel_set_func (GtkTreeViewColumn *tree_column,
130 GtkCellRenderer *cell,
135 ButtonEntry *button_entry;
137 gtk_tree_model_get (model, iter,
138 BUTTONENTRY_COLUMN, &button_entry,
141 if (button_entry == NULL) {
142 g_object_set (G_OBJECT (cell),
146 g_object_set (G_OBJECT (cell),
149 "scancode", button_entry->scancode,
150 "style", PANGO_STYLE_NORMAL,
156 cb_key_edited(GtkCellRendererText *cell, const char *path_string,
157 guint scancode, gpointer data)
159 GtkTreePath *path = gtk_tree_path_new_from_string(path_string);
161 ButtonEntry *button_entry;
163 gtk_tree_model_get_iter(GTK_TREE_MODEL(keys_store), &iter, path);
164 gtk_tree_model_get(GTK_TREE_MODEL(keys_store), &iter,
165 BUTTONENTRY_COLUMN, &button_entry,
168 g_debug("Setting scancode for button %s to %u\n",
169 button_entry->name, scancode);
170 gconf_client_set_int(gcc, button_entry->gconf_key, scancode, NULL);
172 button_entry->scancode = scancode;
173 gtk_tree_model_row_changed(GTK_TREE_MODEL(keys_store), path, &iter);
175 gtk_tree_path_free(path);
179 cb_key_cleared(GtkCellRendererText *cell, const char *path_string,
182 GtkTreePath *path = gtk_tree_path_new_from_string(path_string);
184 ButtonEntry *button_entry;
186 gtk_tree_model_get_iter(GTK_TREE_MODEL(keys_store), &iter, path);
187 gtk_tree_path_free(path);
188 gtk_tree_model_get(GTK_TREE_MODEL(keys_store), &iter,
189 BUTTONENTRY_COLUMN, &button_entry,
192 g_debug("Clearing scancode for button %s\n", button_entry->name);
193 gconf_client_set_int(gcc, button_entry->gconf_key, 0, NULL);
194 // prefer 0 value over unset key.
196 button_entry->scancode = 0;
199 static void cb_combo_changed(GtkComboBox * widget, gpointer data)
202 gconf_client_set_int(gcc, kGConfMapping,
203 gtk_combo_box_get_active(combo), NULL);
206 static void cb_dialog_response(GtkWidget * button, gpointer data)
208 gtk_widget_destroy(GTK_WIDGET(dialog));
211 void controls_setup()
213 GConfValue* mapping = gconf_client_get(gcc, kGConfMapping, NULL);
216 // Key not set; setup defaults
218 for (i = 0; buttons[i].name; i++) {
219 gconf_client_set_int(gcc,
220 buttons[i].gconf_key, buttons[i].default_scancode, NULL);
223 g_debug("Loading default key mappings\n");
225 gconf_client_set_int(gcc, kGConfMapping, 1, NULL);
227 // If this is set, consider defaults loaded.
228 gconf_value_free(mapping);
232 void controls_dialog(GtkWindow* parent)
234 dialog = GTK_DIALOG(gtk_dialog_new_with_buttons("Controls",
235 parent, GTK_DIALOG_MODAL,
236 GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE, NULL));
238 combo = GTK_COMBO_BOX(gtk_combo_box_new_text());
239 gtk_combo_box_append_text(combo, "No controls/Use config file");
240 gtk_combo_box_append_text(combo, "Use physical keys");
241 gtk_combo_box_append_text(combo, "Use touchscreen");
243 none_label = GTK_LABEL(gtk_label_new("Check documentation for details."));
245 keys_scroll = GTK_SCROLLED_WINDOW(gtk_scrolled_window_new(NULL, NULL));
246 gtk_scrolled_window_set_policy(keys_scroll,
247 GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
248 keys_store = GTK_LIST_STORE(gtk_list_store_new(N_COLUMNS,
249 G_TYPE_STRING, G_TYPE_POINTER));
250 keys_list = GTK_TREE_VIEW(
251 gtk_tree_view_new_with_model(GTK_TREE_MODEL(keys_store)));
253 GtkCellRenderer* renderer = GTK_CELL_RENDERER(gtk_cell_renderer_text_new());
254 GtkTreeViewColumn * column =
255 gtk_tree_view_column_new_with_attributes ("Button",
256 gtk_cell_renderer_text_new(),
257 "text", BUTTON_COLUMN,
259 gtk_tree_view_column_set_resizable(column, FALSE);
260 gtk_tree_view_column_set_expand(column, TRUE);
261 gtk_tree_view_append_column(keys_list, column);
263 renderer = GTK_CELL_RENDERER(cell_renderer_key_new());
264 column = gtk_tree_view_column_new_with_attributes("Key", renderer, NULL);
265 gtk_tree_view_column_set_cell_data_func(column, renderer, accel_set_func, NULL, NULL);
266 gtk_tree_view_column_set_resizable(column, FALSE);
267 gtk_tree_view_column_set_min_width(column, 240);
268 gtk_tree_view_append_column(keys_list, column);
269 gtk_tree_view_set_headers_visible(keys_list, TRUE);
272 for (i = 0; buttons[i].name; i++) {
274 gtk_list_store_append(keys_store, &iter);
275 gtk_list_store_set(keys_store, &iter,
276 BUTTON_COLUMN, buttons[i].name,
277 BUTTONENTRY_COLUMN, &buttons[i],
281 ts_label = GTK_LABEL(gtk_label_new("Not implemented."));
283 gtk_window_resize(GTK_WINDOW(dialog), 600, 340);
284 gtk_box_pack_start(GTK_BOX(dialog->vbox), GTK_WIDGET(combo),
285 FALSE, FALSE, HILDON_MARGIN_HALF);
286 gtk_container_add(GTK_CONTAINER(keys_scroll), GTK_WIDGET(keys_list));
287 gtk_box_pack_start_defaults(GTK_BOX(dialog->vbox), GTK_WIDGET(none_label));
288 gtk_box_pack_start_defaults(GTK_BOX(dialog->vbox), GTK_WIDGET(keys_scroll));
289 gtk_box_pack_start_defaults(GTK_BOX(dialog->vbox), GTK_WIDGET(ts_label));
293 g_signal_connect(G_OBJECT(dialog), "response",
294 G_CALLBACK (cb_dialog_response), NULL);
295 g_signal_connect(G_OBJECT(combo), "changed",
296 G_CALLBACK(cb_combo_changed), NULL);
297 g_signal_connect(G_OBJECT(renderer), "accel_edited",
298 G_CALLBACK(cb_key_edited), NULL);
299 g_signal_connect(G_OBJECT(renderer), "accel_cleared",
300 G_CALLBACK(cb_key_cleared), NULL);
302 gtk_widget_show(GTK_WIDGET(dialog));