From 6a6e03d5963a6337e2e091cfed6c3ee3e63842be Mon Sep 17 00:00:00 2001 From: Nikias Bassen Date: Tue, 15 Dec 2009 14:33:11 +0100 Subject: Initial commit. --- src/Makefile.am | 9 + src/dock.png | Bin 0 -> 15852 bytes src/sbmanager.c | 561 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 570 insertions(+) create mode 100644 src/Makefile.am create mode 100644 src/dock.png create mode 100644 src/sbmanager.c (limited to 'src') diff --git a/src/Makefile.am b/src/Makefile.am new file mode 100644 index 0000000..7529a0e --- /dev/null +++ b/src/Makefile.am @@ -0,0 +1,9 @@ +AM_CFLAGS = $(GLOBAL_CFLAGS) $(libiphone_CFLAGS) $(libglib2_CFLAGS) $(libgthread2_CFLAGS) $(libplist_CFLAGS) $(libclutter_CFLAGS) $(libcluttergtk_CFLAGS) $(libgtk_CFLAGS) $(libgdkpixbuf_CFLAGS) +AM_LDFLAGS = $(libiphone_LIBS) $(libglib2_LIBS) $(libgthread2_LIBS) $(libplist_LIBS) $(libclutter_LIBS) $(libcluttergtk_LIBS) $(libgtk_LIBS) $(libgdkpixbuf_LIBS) + +bin_PROGRAMS = sbmanager + +sbmanager_SOURCES = sbmanager.c +sbmanager_CFLAGS = $(AM_CFLAGS) +sbmanager_LDFLAGS = $(AM_LDFLAGS) + diff --git a/src/dock.png b/src/dock.png new file mode 100644 index 0000000..9e4d49c Binary files /dev/null and b/src/dock.png differ diff --git a/src/sbmanager.c b/src/sbmanager.c new file mode 100644 index 0000000..ad17a92 --- /dev/null +++ b/src/sbmanager.c @@ -0,0 +1,561 @@ +/* +sbmanager -- Manage iPhone/iPod Touch SpringBoard icons from your computer! + +Copyright (C) 2009 Nikias Bassen + +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; either version 2 of the License, or +(at your option) any later version. + +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, write to the Free Software +Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA +*/ + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#define ITEM_FONT "Sans Bold 7" + +typedef struct { + GtkWidget *window; +} SBManagerApp; + +typedef struct { + plist_t node; + ClutterActor *texture; + ClutterActor *label; + gboolean is_dock_item; +} SBItem; + +ClutterActor *stage = NULL; +ClutterActor *clock_label = NULL; +ClutterColor text_color = {255, 255, 255, 255}; + +GMutex *selected_mutex = NULL; +ClutterActor *selected = NULL; +SBItem *selected_item = NULL; + +gfloat start_x = 0.0; +gfloat start_y = 0.0; + +GList *dockitems = NULL; +GList *sbpages = NULL; + +GList *this_page = NULL; +int current_page = 0; + +static void sbitem_free(SBItem *a) +{ + if (a) { + if (a->node) { + plist_free(a->node); + } + if (a->texture) { + free(a->texture); + } + } +} + +static void sbpage_free(GList *sbitems) +{ + if (sbitems) { + g_list_foreach(sbitems, (GFunc)(sbitem_free), NULL); + g_list_free(sbitems); + } +} + +static void get_icon_for_node(plist_t node, GList **list, sbservices_client_t sbc) +{ + char *png = NULL; + uint64_t pngsize = 0; + SBItem *di = NULL; + plist_t valuenode = NULL; + if (plist_get_node_type(node) != PLIST_DICT) { + di = g_new0(SBItem, 1); + *list = g_list_append(*list, di); + return; + } + valuenode = plist_dict_get_item(node, "displayIdentifier"); + if (valuenode && (plist_get_node_type(valuenode) == PLIST_STRING)) { + char *value = NULL; + plist_get_string_val(valuenode, &value); + printf("retrieving icon for '%s'\n", value); + if ((sbservices_get_icon_pngdata(sbc, value, &png, &pngsize) == SBSERVICES_E_SUCCESS) && (pngsize > 0)) { + FILE *f = fopen("/tmp/temp.png", "w"); + GError *err = NULL; + fwrite(png, 1, pngsize, f); + fclose(f); + ClutterActor *actor = clutter_texture_new_from_file("/tmp/temp.png", &err); + if (actor) { + plist_t nn = plist_dict_get_item(node, "displayName"); + di = g_new0(SBItem, 1); + di->node = plist_copy(node); + di->texture = actor; + if (nn && (plist_get_node_type(nn) == PLIST_STRING)) { + char *txtval = NULL; + plist_get_string_val(nn, &txtval); + actor = clutter_text_new_full(ITEM_FONT, txtval, &text_color); + di->label = actor; + } + *list = g_list_append(*list, di); + } + if (err) { + fprintf(stderr, "ERROR: %s\n", err->message); + g_error_free(err); + } + } else { + fprintf(stderr, "ERROR: could not get icon for '%s'\n", value); + } + free(value); + } +} + +static void get_icons(const char *uuid) +{ + iphone_device_t phone = NULL; + lockdownd_client_t client = NULL; + sbservices_client_t sbc = NULL; + int port = 0; + plist_t iconstate = NULL; + int total; + + if (sbpages) { + g_list_foreach(sbpages, (GFunc)(sbpage_free), NULL); + g_list_free(sbpages); + sbpages = NULL; + } + if (dockitems) { + sbpage_free(dockitems); + dockitems = NULL; + } + + if (IPHONE_E_SUCCESS != iphone_device_new(&phone, uuid)) { + fprintf(stderr, "No iPhone found, is it plugged in?\n"); + return; + } + + if (LOCKDOWN_E_SUCCESS != lockdownd_client_new(phone, &client)) { + fprintf(stderr, "Could not connect to lockdownd. Exiting.\n"); + goto leave_cleanup; + } + + if ((lockdownd_start_service(client, "com.apple.springboardservices", &port) != LOCKDOWN_E_SUCCESS) || !port) { + fprintf(stderr, "Could not start com.apple.springboardservices service! Remind that this feature is only supported in OS 3.1 and later!\n"); + goto leave_cleanup; + } + if (sbservices_client_new(phone, port, &sbc) != SBSERVICES_E_SUCCESS) { + fprintf(stderr, "Could not connect to springboardservices!\n"); + goto leave_cleanup; + } + if (sbservices_get_icon_state(sbc, &iconstate) != SBSERVICES_E_SUCCESS || !iconstate) { + fprintf(stderr, "ERROR: Could not get icon state!\n"); + goto leave_cleanup; + } + if (plist_get_node_type(iconstate) != PLIST_ARRAY) { + fprintf(stderr, "ERROR: icon state is not an array as expected!\n"); + goto leave_cleanup; + } + total = plist_array_get_size(iconstate); + + if (total < 1) { + fprintf(stderr, "ERROR: No icons returned in icon state\n"); + goto leave_cleanup; + } else { + int i; + int count; + plist_t dock = plist_array_get_item(iconstate, 0); + if ((plist_get_node_type(dock) != PLIST_ARRAY) || (plist_array_get_size(dock) < 1)) { + fprintf(stderr, "ERROR: error getting outer dock icon array!\n"); + goto leave_cleanup; + } + dock = plist_array_get_item(dock, 0); + if (plist_get_node_type(dock) != PLIST_ARRAY) { + fprintf(stderr, "ERROR: error getting inner dock icon array!\n"); + goto leave_cleanup; + } + count = plist_array_get_size(dock); + for (i = 0; i < count; i++) { + plist_t node = plist_array_get_item(dock, i); + get_icon_for_node(node, &dockitems, sbc); + } + if (total > 1) { + /* get all icons for the other pages */ + int p,r,rows; + for (p = 1; p < total; p++) { + plist_t npage = plist_array_get_item(iconstate, p); + GList *page = NULL; + if ((plist_get_node_type(npage) != PLIST_ARRAY) || (plist_array_get_size(npage) < 1)) { + fprintf(stderr, "ERROR: error getting outer page icon array!\n"); + goto leave_cleanup; + } + rows = plist_array_get_size(npage); + for (r = 0; r < rows; r++) { + printf("page %d, row %d\n", p, r); + + plist_t nrow = plist_array_get_item(npage, r); + if (plist_get_node_type(nrow) != PLIST_ARRAY) { + fprintf(stderr, "ERROR: error getting page row icon array!\n"); + goto leave_cleanup; + } + count = plist_array_get_size(nrow); + for (i = 0; i < count; i++) { + plist_t node = plist_array_get_item(nrow, i); + get_icon_for_node(node, &page, sbc); + } + } + if (page) { + sbpages = g_list_append(sbpages, page); + } + } + } + } + +leave_cleanup: + if (iconstate) { + plist_free(iconstate); + } + if (sbc) { + sbservices_client_free(sbc); + } + if (client) { + lockdownd_client_free(client); + } + iphone_device_free(phone); + + return; +} + +static void clock_cb (ClutterTimeline *timeline, gint msecs, SBManagerApp *app) +{ + time_t t = time(NULL); + struct tm *curtime = localtime(&t); + gchar *ctext = g_strdup_printf("%02d:%02d", curtime->tm_hour, curtime->tm_min); + clutter_text_set_text(CLUTTER_TEXT(clock_label), ctext); + clutter_actor_set_position(clock_label, (clutter_actor_get_width(stage)-clutter_actor_get_width(clock_label)) / 2, 2); + g_free(ctext); +} + +static gboolean item_button_press (ClutterActor *actor, ClutterButtonEvent *event, gpointer user_data) +{ + if (!user_data) { + return FALSE; + } + + SBItem *item = (SBItem*)user_data; + + char *strval = NULL; + plist_t node = plist_dict_get_item(item->node, "displayName"); + if (node && plist_get_node_type(node) == PLIST_STRING) { + plist_get_string_val(node, &strval); + } + + g_mutex_lock(selected_mutex); + printf("%s: %s mouse pressed\n", __func__, strval); + + if (actor) { + ClutterActor *sc = clutter_actor_get_parent(actor); + if (item->is_dock_item) { + GList *children = clutter_container_get_children(CLUTTER_CONTAINER(sc)); + if (children) { + ClutterActor *icon = g_list_nth_data(children, 0); + ClutterActor *label = g_list_nth_data(children, 1); + clutter_actor_set_y(label, clutter_actor_get_y(icon) + 62.0); + g_list_free(children); + } + } + clutter_actor_set_scale_full(sc, 1.2, 1.2, clutter_actor_get_x(actor) + clutter_actor_get_width(actor)/2, clutter_actor_get_y(actor) + clutter_actor_get_height(actor)/2); + clutter_actor_raise_top(sc); + clutter_actor_set_opacity(sc, 160); + selected = sc; + selected_item = item; + start_x = event->x; + start_y = event->y; + } + g_mutex_unlock(selected_mutex); + + return TRUE; +} + +static gboolean item_button_release (ClutterActor *actor, ClutterButtonEvent *event, gpointer user_data) +{ + if (!user_data) { + return FALSE; + } + + SBItem *item = (SBItem*)user_data; + char *strval = NULL; + plist_t node = plist_dict_get_item(item->node, "displayName"); + if (node && plist_get_node_type(node) == PLIST_STRING) { + plist_get_string_val(node, &strval); + } + + g_mutex_lock(selected_mutex); + printf("%s: %s mouse released\n", __func__, strval); + + if (actor) { + ClutterActor *sc = clutter_actor_get_parent(actor); + if (item->is_dock_item) { + GList *children = clutter_container_get_children(CLUTTER_CONTAINER(sc)); + if (children) { + ClutterActor *icon = g_list_nth_data(children, 0); + ClutterActor *label = g_list_nth_data(children, 1); + clutter_actor_set_y(label, clutter_actor_get_y(icon) + 69.0); + g_list_free(children); + } + } + clutter_actor_set_scale_full(sc, 1.0, 1.0, clutter_actor_get_x(actor) + clutter_actor_get_width(actor)/2, clutter_actor_get_y(actor) + clutter_actor_get_height(actor)/2); + clutter_actor_set_opacity(sc, 255); + } + + selected = NULL; + selected_item = NULL; + start_x = 0.0; + start_y = 0.0; + + g_mutex_unlock(selected_mutex); + + return TRUE; +} + +static void redraw_icons(SBManagerApp *app) +{ + guint i; + gfloat ypos; + gfloat xpos; + + if (dockitems) { + ypos = 398.0; + xpos = 16.0; + printf("%s: drawing dock icons\n", __func__); + for (i = 0; i < g_list_length(dockitems); i++) { + SBItem *item = (SBItem*)g_list_nth_data(dockitems, i); + if (item && item->texture && item->node) { + item->is_dock_item = TRUE; + ClutterActor *grp = clutter_group_new(); + ClutterActor *actor = item->texture; + clutter_container_add_actor(CLUTTER_CONTAINER(grp), actor); + clutter_actor_set_position(actor, xpos, ypos); + clutter_actor_set_reactive(actor, TRUE); + g_signal_connect(actor, "button-press-event", G_CALLBACK (item_button_press), item); + g_signal_connect(actor, "button-release-event", G_CALLBACK (item_button_release), item); + clutter_actor_show(actor); + actor = item->label; + clutter_actor_set_position(actor, xpos+(59.0 - clutter_actor_get_width(actor))/2, ypos+69.0); + clutter_actor_show(actor); + clutter_container_add_actor(CLUTTER_CONTAINER(grp), actor); + clutter_container_add_actor(CLUTTER_CONTAINER(stage), grp); + } + xpos += 76; + } + } + clutter_stage_ensure_redraw(CLUTTER_STAGE(stage)); + if (sbpages) { + ypos = 32.0; + xpos = 16.0; + printf("%s: %d pages\n", __func__, g_list_length(sbpages)); + printf("%s: drawing page icons for page %d\n", __func__, current_page); + this_page = g_list_nth_data(sbpages, current_page); + for (i = 0; i < g_list_length(this_page); i++) { + SBItem *item = (SBItem*)g_list_nth_data(this_page, i); + if (item && item->texture && item->node) { + item->is_dock_item = FALSE; + ClutterActor *grp = clutter_group_new(); + ClutterActor *actor = item->texture; + clutter_container_add_actor(CLUTTER_CONTAINER(grp), actor); + clutter_actor_set_position(actor, xpos, ypos); + clutter_actor_set_reactive(actor, TRUE); + g_signal_connect(actor, "button-press-event", G_CALLBACK (item_button_press), item); + g_signal_connect(actor, "button-release-event", G_CALLBACK (item_button_release), item); + clutter_actor_show(actor); + actor = item->label; + clutter_actor_set_position(actor, xpos+(59.0 - clutter_actor_get_width(actor))/2, ypos+62.0); + clutter_actor_show(actor); + clutter_container_add_actor(CLUTTER_CONTAINER(grp), actor); + clutter_container_add_actor(CLUTTER_CONTAINER(stage), grp); + } + if (((i+1) % 4) == 0) { + xpos = 16.0; + ypos += 88.0; + } else { + xpos += 76.0; + } + } + } + clutter_stage_ensure_redraw(CLUTTER_STAGE(stage)); +} + +static gboolean stage_motion (ClutterActor *actor, ClutterMotionEvent *event, gpointer user_data) +{ + /* check if an item has been raised */ + if (!selected || !selected_item) { + return FALSE; + } + +/* gfloat oldx = clutter_actor_get_x(selected); + gfloat oldy = clutter_actor_get_y(selected); + + clutter_actor_set_position(selected, oldx + (event->x - start_x), oldy + (event->y - start_y)); +*/ + clutter_actor_move_by(selected, (event->x - start_x), event->y - start_y); + + start_x = event->x; + start_y = event->y; + + if (selected_item->is_dock_item) { + printf("an icon from the dock is moving\n"); + } else { + printf("a regular icon is moving\n"); + } + + return TRUE; +} + +static gboolean form_map(GtkWidget *widget, GdkEvent *event, SBManagerApp *app) +{ + printf("%s: mapped\n", __func__); + clutter_stage_ensure_redraw(CLUTTER_STAGE(stage)); + + redraw_icons(app); + + clutter_stage_ensure_redraw(CLUTTER_STAGE(stage)); + + return TRUE; +} + +static gboolean form_focus_change(GtkWidget *widget, GdkEventFocus *event, gpointer user_data) +{ + if (!user_data) { + return FALSE; + } + + ClutterActor *actor = (ClutterActor*)user_data; + + if (event->in) { + clutter_timeline_start(CLUTTER_TIMELINE(actor)); + } else { + clutter_timeline_pause(CLUTTER_TIMELINE(actor)); + } + + return TRUE; +} + +int main(int argc, char **argv) +{ + SBManagerApp *app; + ClutterTimeline *timeline; + ClutterColor stage_color = { 0x00, 0x00, 0x00, 0xff }; /* Black */ + ClutterActor *actor; + + app = g_new0 (SBManagerApp, 1); + if (!app) { + printf("Error: out of memory!\n"); + return -1; + } + + if (gtk_clutter_init (&argc, &argv) != CLUTTER_INIT_SUCCESS) { + g_error ("Unable to initialize GtkClutter"); + } + + if (!g_thread_supported()) + g_thread_init(NULL); + + get_icons(NULL); + + /* Create the window and some child widgets: */ + app->window = gtk_window_new (GTK_WINDOW_TOPLEVEL); + gtk_window_set_title (GTK_WINDOW(app->window), "SpringBoard Manager"); + GtkWidget *vbox = gtk_vbox_new (FALSE, 6); + gtk_container_add (GTK_CONTAINER (app->window), vbox); + gtk_widget_show (vbox); + GtkWidget *button = gtk_button_new_with_label ("Upload changes to device"); + gtk_box_pack_end (GTK_BOX (vbox), button, FALSE, FALSE, 0); + gtk_widget_show (button); + + /* Stop the application when the window is closed: */ + g_signal_connect (app->window, "hide", G_CALLBACK (gtk_main_quit), app); + + /* Create the clutter widget: */ + GtkWidget *clutter_widget = gtk_clutter_embed_new (); + gtk_box_pack_start (GTK_BOX (vbox), clutter_widget, TRUE, TRUE, 0); + gtk_widget_show (clutter_widget); + + /* Set the size of the widget, because we should not set the size of its + * stage when using GtkClutterEmbed. + */ + gtk_widget_set_size_request (clutter_widget, 320, 480); + + /* Get the stage and set its size and color: */ + stage = gtk_clutter_embed_get_stage (GTK_CLUTTER_EMBED (clutter_widget)); + + clutter_stage_set_color (CLUTTER_STAGE (stage), &stage_color); + + /* dock background */ + GError *err = NULL; + actor = clutter_texture_new_from_file("./dock.png", &err); + if (err) { + g_error_free(err); + } + if (actor) { + clutter_actor_set_position(actor, 0, clutter_actor_get_height(stage) - clutter_actor_get_height(actor)); + clutter_actor_show(actor); + clutter_group_add (CLUTTER_GROUP(stage), actor); + } else { + fprintf(stderr, "could not load dock.png\n"); + } + + /* clock widget */ + actor = clutter_text_new_full ("Sans Bold 9", "00:00\nblah", &text_color); + gint xpos = (clutter_actor_get_width(stage)-clutter_actor_get_width(actor))/2; + clutter_actor_set_position(actor, xpos, 2); + clutter_actor_show(actor); + clutter_group_add (CLUTTER_GROUP (stage), actor); + clock_label = actor; + + /* Show the stage: */ + clutter_actor_show (stage); + + /* Create a timeline to manage animation */ + timeline = clutter_timeline_new (200); + clutter_timeline_set_loop(timeline, TRUE); /* have it loop */ + + /* fire a callback for frame change */ + g_signal_connect(timeline, "completed", G_CALLBACK (clock_cb), app); + + /* and start it */ + clutter_timeline_start (timeline); + + /* Show the window: */ + gtk_widget_show_all (GTK_WIDGET (app->window)); + + g_signal_connect(stage, "motion-event", G_CALLBACK (stage_motion), app); + + g_signal_connect( G_OBJECT(app->window), "map-event", G_CALLBACK (form_map), app); + + g_signal_connect( G_OBJECT(app->window), "focus-in-event", G_CALLBACK (form_focus_change), timeline); + g_signal_connect( G_OBJECT(app->window), "focus-out-event", G_CALLBACK (form_focus_change), timeline); + + selected_mutex = g_mutex_new(); + + /* Start the main loop, so we can respond to events: */ + gtk_main (); + + return 0; +} -- cgit v1.1-32-gdbae