* ported to maemo by Thomas Thurman, 2009
* suggestions welcome
* Compile with:
- * gcc -Wall -g rfk.c -o rfk `pkg-config --cflags --libs gtk+-2.0 hildon-1`
+ * gcc -Wall -g rfk.c -o rfk `pkg-config --cflags --libs gtk+-2.0 hildon-1 dbus-glib-1 dbus-1`
*/
+#include <dbus/dbus-glib.h>
#include <gtk/gtk.h>
#include <stdlib.h>
#include <glib.h>
const int amount_of_random_stuff = 15;
+typedef enum {
+ STATE_PROLOGUE,
+ STATE_PLAYING,
+ STATE_EPILOGUE,
+ STATE_LAST
+} StateOfPlay;
+
+StateOfPlay current_state = STATE_LAST;
+GtkWidget* state_widget[STATE_LAST];
+
GSList *nki = NULL;
guint nki_count = 0;
GtkWidget *arena[ARENA_WIDTH][ARENA_HEIGHT];
-GtkWidget *intro, *table, *window, *robot, *kitten;
+GtkWidget *window, *robot, *kitten;
int robot_x, robot_y;
gboolean *used = NULL;
GdkPixbuf *robot_pic, *love_pic, *kitten_pic;
-GtkWidget *animation_area;
const GdkColor black = { 0, };
r = random() % nki_count;
}
while (used[r]);
-
used[r] = TRUE;
+
return g_slist_nth_data (nki, r);
}
{
arena[x][y] = item;
- gtk_table_attach_defaults (GTK_TABLE (table),
+ gtk_table_attach_defaults (GTK_TABLE (state_widget[STATE_PLAYING]),
item,
x, x+1,
y, y+1);
{
HildonNote* note = HILDON_NOTE
(hildon_note_new_information (GTK_WINDOW (window),
- message));
+ message?message:
+ "Some message was supposed to be here."));
gtk_dialog_run (GTK_DIALOG (note));
gtk_widget_destroy (GTK_WIDGET (note));
}
}
fclose (nki_file);
-
- used = g_malloc0 (nki_count);
}
void
}
/****************************************************************/
+/* Stop doing that, and do something else. */
+/****************************************************************/
+static void
+switch_state (StateOfPlay new_state)
+{
+ if (current_state != STATE_LAST)
+ {
+ gtk_container_remove (GTK_CONTAINER (window), state_widget[current_state]);
+ }
+ gtk_container_add (GTK_CONTAINER (window), state_widget[new_state]);
+
+ gtk_widget_show_all (window);
+ gdk_window_set_events (GTK_WIDGET (window)->window,
+ gdk_window_get_events(GTK_WIDGET (window)->window) | GDK_BUTTON_PRESS_MASK);
+
+ current_state = new_state;
+}
+
+/****************************************************************/
/* The ending animation. */
/****************************************************************/
+gboolean animation_running = FALSE;
+
+/*
+static gboolean
+love_animation_draw
+*/
static gboolean
ending_animation_quit (gpointer data)
{
- gtk_main_quit ();
+ switch_state (STATE_PROLOGUE);
return FALSE;
}
static gboolean
ending_animation_draw (GtkWidget *widget, GdkEventExpose *event, gpointer data)
{
- /* We only run through once, so just make it static. */
static int cycle_count = 0;
static int robot_x = 0;
static int robot_stop = 0;
static int kitten_x = 0;
static int all_y = 0;
+ static GdkGC *gc = NULL;
const int stepsize = 3;
+ static int love_size = 40;
if (!kitten_x)
{
+ gc = gdk_gc_new (GDK_DRAWABLE (widget->window));
+
all_y = (event->area.height - gdk_pixbuf_get_height (love_pic)) / 2;
robot_stop = gdk_pixbuf_get_width (robot_pic) + gdk_pixbuf_get_width (love_pic);
kitten_x = event->area.width - (cycle_count*stepsize + gdk_pixbuf_get_width (kitten_pic));
}
- gdk_gc_set_foreground (widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
- &black);
+ gdk_gc_set_foreground (gc, &black);
- gdk_draw_rectangle (GDK_DRAWABLE(widget->window),
- widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
+ gdk_draw_rectangle (GDK_DRAWABLE (widget->window),
+ gc,
TRUE,
0, 0, event->area.width, event->area.height);
- gdk_draw_pixbuf (GDK_DRAWABLE(widget->window),
- widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
+ gdk_draw_pixbuf (GDK_DRAWABLE (widget->window),
+ gc,
robot_pic, 0, 0,
robot_x, all_y,
-1, -1,
GDK_RGB_DITHER_NONE, 0, 0);
- gdk_draw_pixbuf (GDK_DRAWABLE(widget->window),
- widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
+ gdk_draw_pixbuf (GDK_DRAWABLE (widget->window),
+ gc,
kitten_pic, 0, 0,
kitten_x, all_y,
-1, -1,
GDK_RGB_DITHER_NONE, 0, 0);
- cycle_count++;
- robot_x += stepsize;
- kitten_x -= stepsize;
-
- if (robot_x+robot_stop >= kitten_x)
+ if (robot_x+robot_stop < kitten_x)
{
+ cycle_count++;
+ robot_x += stepsize;
+ kitten_x -= stepsize;
+ }
+ else
+ {
+ GdkPixbuf *scaled_love_pic =
+ gdk_pixbuf_scale_simple (love_pic,
+ love_size,
+ love_size,
+ GDK_INTERP_BILINEAR);
+
gdk_draw_pixbuf (GDK_DRAWABLE(widget->window),
- widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
- love_pic, 0, 0,
+ gc,
+ scaled_love_pic, 0, 0,
robot_x + gdk_pixbuf_get_width (robot_pic), all_y,
-1, -1,
GDK_RGB_DITHER_NONE, 0, 0);
- g_object_unref (love_pic);
- love_pic = NULL;
+ love_size ++;
- g_timeout_add (2000, ending_animation_quit, NULL);
+ if (love_size >= gdk_pixbuf_get_width (love_pic))
+ {
+ animation_running = FALSE;
+
+ g_timeout_add (2000, ending_animation_quit, NULL);
+
+ gdk_gc_unref (gc);
+ love_size = 40;
+ cycle_count = 0;
+ robot_x = 0;
+ robot_stop = 0;
+ kitten_x = 0;
+ all_y = 0;
+ gc = NULL;
+ }
}
return TRUE;
static gboolean
ending_animation_step (gpointer data)
{
- if (love_pic)
+ if (animation_running)
{
- gdk_window_invalidate_rect (animation_area->window,
+ gdk_window_invalidate_rect (state_widget[STATE_EPILOGUE]->window,
NULL, TRUE);
return TRUE;
static void
ending_animation ()
{
- animation_area = gtk_drawing_area_new ();
-
- gtk_container_remove (GTK_CONTAINER (window), GTK_WIDGET (table));
- gtk_container_add (GTK_CONTAINER (window), GTK_WIDGET (animation_area));
- gtk_widget_show_all (window);
-
- g_signal_connect (G_OBJECT (animation_area),
- "expose_event", G_CALLBACK (ending_animation_draw), NULL);
- g_timeout_add (10, ending_animation_step, NULL);
+ if (current_state!=STATE_EPILOGUE)
+ {
+ animation_running = TRUE;
+ g_timeout_add (10, ending_animation_step, NULL);
+ }
}
/****************************************************************/
if (new_space == kitten)
{
- ending_animation ();
+ switch_state (STATE_EPILOGUE);
}
return TRUE;
g_object_ref (new_space);
- gtk_container_remove (GTK_CONTAINER (table), robot);
- gtk_container_remove (GTK_CONTAINER (table), new_space);
+ gtk_container_remove (GTK_CONTAINER (state_widget[STATE_PLAYING]), robot);
+ gtk_container_remove (GTK_CONTAINER (state_widget[STATE_PLAYING]), new_space);
place_in_arena_at_xy (new_space, robot_x, robot_y);
place_in_arena_at_xy (robot, robot_x+dx, robot_y+dy);
int rx, ry;
double angle;
+ if (current_state!=STATE_PLAYING)
+ {
+ return TRUE;
+ }
+
rx = (robot->allocation.x+robot->allocation.width/2);
ry = (robot->allocation.y+robot->allocation.height/2);
gint i;
guint keyval = event->keyval;
+ if (current_state!=STATE_PLAYING)
+ {
+ return FALSE;
+ }
+
if (keyval>='A' && keyval<='Z')
{
keyval += ('a'-'A');
return FALSE;
}
-void
-create_window (void)
+/****************************************************************/
+/* Online help. */
+/****************************************************************/
+gboolean
+get_help (gpointer button, gpointer data)
{
- window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
- gtk_window_set_title (GTK_WINDOW (window), "robotfindskitten");
- gtk_widget_modify_bg (window, GTK_STATE_NORMAL, &black);
+ DBusGConnection *connection;
+ GError *error = NULL;
+
+ DBusGProxy *proxy;
+
+ connection = dbus_g_bus_get (DBUS_BUS_SESSION,
+ &error);
+ if (connection == NULL)
+ {
+ show_message (error->message);
+ g_error_free (error);
+ return FALSE;
+ }
+
+ proxy = dbus_g_proxy_new_for_name (connection,
+ "com.nokia.osso_browser",
+ "/com/nokia/osso_browser/request",
+ "com.nokia.osso_browser");
+
+ error = NULL;
+ if (!dbus_g_proxy_call (proxy, "load_url", &error,
+ G_TYPE_STRING, "/usr/share/rfk/help.html",
+ G_TYPE_INVALID,
+ G_TYPE_INVALID))
+ {
+ show_message (error->message);
+ g_error_free (error);
+ return FALSE;
+ }
+ return FALSE;
}
void
-set_up_game (void)
+play_game (gpointer button, gpointer data)
{
- guint x, y;
+ switch_state (STATE_PLAYING);
+}
- g_signal_connect (G_OBJECT (window), "button-press-event", G_CALLBACK (on_window_clicked), NULL);
- g_signal_connect (G_OBJECT (window), "key-press-event", G_CALLBACK (on_key_pressed), NULL);
- g_signal_connect (G_OBJECT (window), "delete_event", G_CALLBACK (gtk_main_quit), NULL);
- gdk_window_set_events (GTK_WIDGET (window)->window,
- gdk_window_get_events(GTK_WIDGET (window)->window) | GDK_BUTTON_PRESS_MASK);
-
- table = gtk_table_new (ARENA_HEIGHT, ARENA_WIDTH, TRUE);
- gtk_container_remove (GTK_CONTAINER (window), GTK_WIDGET (intro));
- gtk_container_add (GTK_CONTAINER (window), GTK_WIDGET (table));
+static void
+set_up_board (void)
+{
+ guint x, y;
- robot = gtk_label_new ("#");
- g_object_ref (robot);
- kitten = random_character ("You found kitten! Way to go, robot!");
+ if (current_state==STATE_PLAYING)
+ {
+ /* end of the game; clean up */
- place_in_arena_randomly (robot);
- place_in_arena_randomly (kitten);
+ for (x=0; x < ARENA_WIDTH; x++)
+ for (y=0; y < ARENA_HEIGHT; y++)
+ if (arena[x][y])
+ {
+ gtk_container_remove (GTK_CONTAINER (state_widget[STATE_PLAYING]),
+ arena[x][y]);
+ arena[x][y] = NULL;
+ }
- if (nki_count < amount_of_random_stuff)
- {
- gtk_widget_show_all (window);
- show_message ("There are too few non-kitten items to play a meaningful game.");
- exit (EXIT_FAILURE);
+ g_object_unref (robot);
+ g_object_unref (kitten);
}
+ else
+ {
+ /* make everything new */
+
+ g_free (used);
+ used = g_malloc0 (nki_count * sizeof(gboolean));
- for (x=0; x < amount_of_random_stuff; x++)
- place_in_arena_randomly (random_character (description ()));
+ robot = gtk_label_new ("#");
+ g_object_ref (robot);
+ kitten = random_character ("You found kitten! Way to go, robot!");
+ g_object_ref (kitten);
- for (x=0; x < ARENA_WIDTH; x++)
- for (y=0; y < ARENA_HEIGHT; y++)
- if (!arena[x][y])
- place_in_arena_at_xy (gtk_label_new (NULL), x, y);
+ place_in_arena_randomly (robot);
+ place_in_arena_randomly (kitten);
- gtk_widget_show_all (window);
-}
+ if (nki_count < amount_of_random_stuff)
+ {
+ /* sanity check failed */
+ show_message ("There are too few non-kitten items to play a meaningful game.");
+ exit (EXIT_FAILURE);
+ }
-void
-get_help (gpointer button, gpointer data)
-{
- show_message ("Not yet implemented.");
-}
+ for (x=0; x < amount_of_random_stuff; x++)
+ place_in_arena_randomly (random_character (description ()));
-void
-play_game (gpointer button, gpointer data)
-{
- set_up_game ();
+ for (x=0; x < ARENA_WIDTH; x++)
+ for (y=0; y < ARENA_HEIGHT; y++)
+ if (!arena[x][y])
+ place_in_arena_at_xy (gtk_label_new (NULL), x, y);
+ }
}
-void
-show_intro (void)
+static void
+set_up_widgets (void)
{
GtkWidget *middle = gtk_hbox_new (FALSE, 0);
GtkWidget *buttons = gtk_hbox_new (TRUE, 0);
- GtkWidget *explain = NULL, *help_button, *play_button;
+ GtkWidget *explain = NULL, *help_button, *play_button, *intro;
const char *explanation =
"In this game, you are robot (#). "
"Your job is to find kitten. This task is complicated "
"arrow keys.";
GKeyFile *desktop = g_key_file_new ();
gchar *version;
+ guint x, y;
+
+ /* The window */
+
+ window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
+ gtk_window_set_title (GTK_WINDOW (window), "robotfindskitten");
+ gtk_widget_modify_bg (window, GTK_STATE_NORMAL, &black);
+ g_signal_connect (G_OBJECT (window), "button-press-event", G_CALLBACK (on_window_clicked), NULL);
+ g_signal_connect (G_OBJECT (window), "key-press-event", G_CALLBACK (on_key_pressed), NULL);
+ g_signal_connect (G_OBJECT (window), "delete_event", G_CALLBACK (gtk_main_quit), NULL);
+
+ /* The prologue */
if (g_key_file_load_from_file (desktop,
"/usr/share/applications/hildon/rfk.desktop",
"Play", NULL);
g_signal_connect (play_button, "clicked", G_CALLBACK (play_game), NULL);
- gtk_box_pack_end (buttons, play_button, TRUE, TRUE, 0);
- gtk_box_pack_end (buttons, help_button, TRUE, TRUE, 0);
+ gtk_box_pack_end (GTK_BOX (buttons), play_button, TRUE, TRUE, 0);
+ gtk_box_pack_end (GTK_BOX (buttons), help_button, TRUE, TRUE, 0);
explain = gtk_label_new (explanation);
- gtk_label_set_line_wrap (explain, TRUE);
+ gtk_label_set_line_wrap (GTK_LABEL (explain), TRUE);
- gtk_box_pack_end (middle, explain, TRUE, TRUE, 0);
- gtk_box_pack_end (middle, gtk_image_new_from_pixbuf (robot_pic), FALSE, FALSE, 0);
+ gtk_box_pack_end (GTK_BOX (middle), explain, TRUE, TRUE, 0);
+ gtk_box_pack_end (GTK_BOX (middle), gtk_image_new_from_pixbuf (robot_pic), FALSE, FALSE, 0);
intro = gtk_vbox_new (FALSE, 0);
gtk_box_pack_end (GTK_BOX (intro), buttons, FALSE, FALSE, 0);
gtk_box_pack_end (GTK_BOX (intro), middle, TRUE, TRUE, 0);
gtk_box_pack_end (GTK_BOX (intro), gtk_label_new (version), FALSE, FALSE, 0);
-
g_free (version);
- gtk_container_add (GTK_CONTAINER (window), GTK_WIDGET (intro));
+ state_widget[STATE_PROLOGUE] = intro;
- gtk_widget_show_all (window);
+ /* The game itself */
+
+ state_widget[STATE_PLAYING] = gtk_table_new (ARENA_HEIGHT, ARENA_WIDTH, TRUE);
+ g_signal_connect (state_widget[STATE_PLAYING], "parent-set", G_CALLBACK (set_up_board), NULL);
+
+ for (x=0; x < ARENA_WIDTH; x++)
+ for (y=0; y < ARENA_HEIGHT; y++)
+ arena[x][y] = NULL;
+
+ /* The epilogue */
+ state_widget[STATE_EPILOGUE] = gtk_drawing_area_new ();
+ g_signal_connect (state_widget[STATE_EPILOGUE], "parent-set", G_CALLBACK (ending_animation), NULL);
+ g_signal_connect (G_OBJECT (state_widget[STATE_EPILOGUE]),
+ "expose_event", G_CALLBACK (ending_animation_draw), NULL);
+
+ for (x=0; x<STATE_LAST; x++)
+ {
+ /* so we don't lose them when we take them offscreen */
+ g_object_ref (state_widget[x]);
+ }
}
/****************************************************************/
ensure_messages_loaded ();
load_images ();
- create_window ();
- show_intro ();
-
+ set_up_widgets ();
+ switch_state (STATE_PROLOGUE);
+
gtk_main ();
return EXIT_SUCCESS;