Files
OpenRA/OpenRA.Launcher.Gtk/main.c
2011-01-02 16:11:59 +13:00

318 lines
7.4 KiB
C

/*
* Copyright 2007-2010 The OpenRA Developers (see AUTHORS)
* This file is part of OpenRA, which is free software. It is made
* available to you under the terms of the GNU General Public License
* as published by the Free Software Foundation. For more information,
* see LICENSE.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <gtk/gtk.h>
#include <webkit/webkit.h>
#include "server.h"
#include "bridge.h"
#include "utility.h"
#define WEBSERVER_PORT 48764
GtkWindow * window;
WebKitWebView * browser;
GtkTreeStore * tree_store;
GtkTreeView * tree;
gboolean window_delete(GtkWidget * widget, GdkEvent * event,
gpointer user_data)
{
server_teardown();
gtk_main_quit();
return FALSE;
}
enum
{
ICON_COLUMN,
KEY_COLUMN,
NAME_COLUMN,
N_COLUMNS
};
#define MOD_key_MAX_LEN 16
#define MOD_title_MAX_LEN 32
#define MOD_version_MAX_LEN 16
#define MOD_author_MAX_LEN 32
#define MOD_description_MAX_LEN 128
#define MOD_requires_MAX_LEN 32
#define MAX_NUM_MODS 64
typedef struct mod_t
{
char key[MOD_key_MAX_LEN];
char title[MOD_title_MAX_LEN];
char version[MOD_version_MAX_LEN];
char author[MOD_author_MAX_LEN];
char description[MOD_description_MAX_LEN];
char requires[MOD_requires_MAX_LEN];
int standalone;
} mod_t;
static mod_t mods[MAX_NUM_MODS];
static int mod_count = 0;
typedef void ( * lines_callback ) (char const * line, gpointer data);
//Splits console output into lines and passes each one to a callback
void process_lines(char * const lines, int len, lines_callback cb, gpointer data)
{
int prev = 0, current = 0;
while (current < len)
{
if (lines[current] == '\n')
{
char * line = (char *)malloc(current - prev + 1);
memcpy(line, lines + prev, current - prev);
line[current - prev] = '\0';
cb(line, data);
free(line);
prev = current + 1;
}
current++;
}
}
#define ASSIGN_TO_MOD(FIELD, VAL_OFF) \
strncpy(mod->FIELD, val_start + VAL_OFF, MOD_##FIELD##_MAX_LEN - 1); \
mod->FIELD[MOD_##FIELD##_MAX_LEN - 1] = '\0'
#define min(X, Y) X < Y ? X : Y
void mod_metadata_line(char const * line, gpointer data)
{
mod_t * mod = (mod_t *)data;
char * val_start = strchr(line, ':');
if (memcmp(line, "Mod:", 4) == 0)
{
ASSIGN_TO_MOD(key, 1);
}
else if (memcmp(line, " Title:", min(strlen(line), 8)) == 0)
{
ASSIGN_TO_MOD(title, 2);
}
else if (memcmp(line, " Version:", min(strlen(line), 10)) == 0)
{
ASSIGN_TO_MOD(version, 2);
}
else if (memcmp(line, " Author:", min(strlen(line), 9)) == 0)
{
ASSIGN_TO_MOD(author, 2);
}
else if (memcmp(line, " Description:", min(strlen(line), 14)) == 0)
{
ASSIGN_TO_MOD(description, 2);
}
else if (memcmp(line, " Requires:", min(strlen(line), 11)) == 0)
{
ASSIGN_TO_MOD(requires, 2);
}
else if (memcmp(line, " Standalone:", min(strlen(line), 13)) == 0)
{
if (strcmp(val_start + 2, "True") == 0)
mod->standalone = TRUE;
else
mod->standalone = FALSE;
}
}
mod_t * get_mod(char const * key)
{
int i;
for (i = 0; i < mod_count; i++)
{
if (strcmp(mods[i].key, key) == 0)
return mods + i;
}
return NULL;
}
gboolean find_mod(GtkTreeModel * model, GtkTreePath * path, GtkTreeIter * iter,
gpointer data)
{
mod_t * mod = (mod_t *)data;
gchar * key;
GtkTreeIter new_iter;
gtk_tree_model_get(model, iter, KEY_COLUMN, &key, -1);
if (!key)
{
return FALSE;
}
if (strcmp(mod->requires, key) == 0)
{
gtk_tree_store_append(GTK_TREE_STORE(model), &new_iter, iter);
gtk_tree_store_set(GTK_TREE_STORE(model), &new_iter,
KEY_COLUMN, mod->key,
NAME_COLUMN, mod->title,
-1);
return TRUE;
}
return FALSE;
}
void mod_metadata_callback(GPid pid, gint status, gpointer data)
{
int out_len, * out_fd = (int *)data;
char * msg = NULL;
mod_t mod = mods[mod_count];
GtkTreeIter iter, mod_iter;
mod_count = (mod_count + 1) % MAX_NUM_MODS;
memset(&mod, 0, sizeof(mod_t));
msg = util_get_output(*out_fd, &out_len);
process_lines(msg, out_len, mod_metadata_line, &mod);
free(msg);
gtk_tree_model_get_iter_first(GTK_TREE_MODEL(tree_store), &mod_iter);
if (mod.standalone)
{
gtk_tree_store_append(tree_store, &iter, &mod_iter);
gtk_tree_store_set(tree_store, &iter,
KEY_COLUMN, mod.key,
NAME_COLUMN, mod.title,
-1);
}
else if (!strlen(mod.requires))
{
GtkTreeIter broken_mods_iter;
if (!gtk_tree_model_get_iter_from_string(GTK_TREE_MODEL(tree_store),
&broken_mods_iter, "1"))
{
gtk_tree_store_append(tree_store, &broken_mods_iter, NULL);
}
gtk_tree_store_set(tree_store, &broken_mods_iter,
KEY_COLUMN, mod.key,
NAME_COLUMN, mod.title,
-1);
}
else
{
gtk_tree_model_foreach(GTK_TREE_MODEL(tree_store), find_mod, &mod);
}
close(*out_fd);
free(out_fd);
}
void mod_list_line(char const * mod, gpointer user)
{
util_get_mod_metadata(mod, mod_metadata_callback);
}
void mod_list_callback(GPid pid, gint status, gpointer data)
{
int out_len, * out_fd = (int *)data;
char * msg = NULL;
msg = util_get_output(*out_fd, &out_len);
mod_count = 0;
process_lines(msg, out_len, mod_list_line, NULL);
free(msg);
close(*out_fd);
free(out_fd);
g_spawn_close_pid(pid);
}
void make_tree_view(void)
{
GtkTreeIter iter;
GtkCellRenderer * pixbuf_renderer, * text_renderer;
GtkTreeViewColumn * icon_column, * name_column;
tree_store = gtk_tree_store_new(N_COLUMNS, GDK_TYPE_PIXBUF,
G_TYPE_STRING, G_TYPE_STRING);
gtk_tree_store_append(tree_store, &iter, NULL);
gtk_tree_store_set(tree_store, &iter,
NAME_COLUMN, "MODS",
-1);
tree = GTK_TREE_VIEW(gtk_tree_view_new_with_model(GTK_TREE_MODEL(tree_store)));
g_object_set(tree, "headers-visible", FALSE, NULL);
pixbuf_renderer = gtk_cell_renderer_pixbuf_new();
text_renderer = gtk_cell_renderer_text_new();
icon_column = gtk_tree_view_column_new_with_attributes
("Icon", pixbuf_renderer,
"pixbuf", ICON_COLUMN,
NULL);
gtk_tree_view_column_set_sizing(icon_column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
name_column = gtk_tree_view_column_new_with_attributes
("Name", text_renderer,
"text", NAME_COLUMN,
NULL);
gtk_tree_view_column_set_sizing(name_column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
gtk_tree_view_append_column(tree, icon_column);
gtk_tree_view_append_column(tree, name_column);
}
int main(int argc, char ** argv)
{
GtkWidget * hbox;
server_init(WEBSERVER_PORT);
gtk_init(&argc, &argv);
window = GTK_WINDOW(gtk_window_new(GTK_WINDOW_TOPLEVEL));
gtk_window_set_title(window, "OpenRA Launcher");
gtk_window_set_default_size(window, 800, 600);
browser = WEBKIT_WEB_VIEW(webkit_web_view_new());
g_signal_connect(browser, "window-object-cleared",
G_CALLBACK(bind_js_bridge), 0);
make_tree_view();
util_get_mod_list(mod_list_callback);
hbox = gtk_hbox_new(FALSE, 0);
gtk_widget_set_size_request(GTK_WIDGET(tree), 250, 0);
gtk_box_pack_end(GTK_BOX(hbox), GTK_WIDGET(browser), TRUE, TRUE, 0);
gtk_box_pack_start(GTK_BOX(hbox), GTK_WIDGET(tree), TRUE, TRUE, 0);
gtk_container_add(GTK_CONTAINER(window), hbox);
//TODO: Load the mod html file based on selected mod in launcher
webkit_web_view_load_uri(browser,
"http://localhost:48764/mods/cnc/mod.html");
gtk_widget_show_all(GTK_WIDGET(window));
g_signal_connect(window, "delete-event", G_CALLBACK(window_delete), 0);
gtk_main();
return 0;
}