498 lines
11 KiB
C
498 lines
11 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 <unistd.h>
|
|
|
|
#include "main.h"
|
|
#include "server.h"
|
|
#include "bridge.h"
|
|
#include "utility.h"
|
|
|
|
#define WEBSERVER_PORT 48764
|
|
|
|
GtkWindow * window;
|
|
WebKitWebView * browser;
|
|
GtkTreeStore * tree_store;
|
|
GtkTreeView * tree;
|
|
GdkPixbuf * generic_mod_icon;
|
|
GtkRadioButton * gl_button, * cg_button;
|
|
|
|
static mod_t mods[MAX_NUM_MODS];
|
|
static int mod_count = 0;
|
|
|
|
static int renderer = RENDERER_GL;
|
|
|
|
void free_mod(mod_t * mod)
|
|
{
|
|
g_free(mod->key);
|
|
g_free(mod->title);
|
|
g_free(mod->version);
|
|
g_free(mod->author);
|
|
g_free(mod->description);
|
|
g_free(mod->requires);
|
|
}
|
|
|
|
gboolean window_delete(GtkWidget * widget, GdkEvent * event,
|
|
gpointer user_data)
|
|
{
|
|
int i;
|
|
server_teardown();
|
|
for (i = 0; i < mod_count; i++)
|
|
free_mod(mods + i);
|
|
gtk_main_quit();
|
|
return FALSE;
|
|
}
|
|
|
|
typedef void ( * lines_callback ) (GString const * line, gpointer data);
|
|
|
|
//Splits console output into lines and passes each one to a callback
|
|
void process_lines(GString * lines, lines_callback cb, gpointer data)
|
|
{
|
|
int prev = 0, current = 0;
|
|
while (current < lines->len)
|
|
{
|
|
if (lines->str[current] == '\n')
|
|
{
|
|
GString * line = g_string_new_len(lines->str + prev, current - prev);
|
|
cb(line, data);
|
|
g_string_free(line, TRUE);
|
|
prev = current + 1;
|
|
}
|
|
current++;
|
|
}
|
|
}
|
|
|
|
#define ASSIGN_TO_MOD(FIELD) \
|
|
mod->FIELD = g_strdup(val_start)
|
|
|
|
void mod_metadata_line(GString const * line, gpointer data)
|
|
{
|
|
mod_t * mod = (mod_t *)data;
|
|
gchar * val_start = g_strstr_len(line->str, -1, ":") + 2;
|
|
if (g_str_has_prefix(line->str, "Mod:"))
|
|
{
|
|
ASSIGN_TO_MOD(key);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Title:"))
|
|
{
|
|
ASSIGN_TO_MOD(title);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Version:"))
|
|
{
|
|
ASSIGN_TO_MOD(version);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Author:"))
|
|
{
|
|
ASSIGN_TO_MOD(author);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Description:"))
|
|
{
|
|
ASSIGN_TO_MOD(description);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Requires:"))
|
|
{
|
|
ASSIGN_TO_MOD(requires);
|
|
}
|
|
else if (g_str_has_prefix(line->str, " Standalone:"))
|
|
{
|
|
if (strcmp(val_start, "True") == 0)
|
|
mod->standalone = TRUE;
|
|
else
|
|
mod->standalone = FALSE;
|
|
}
|
|
}
|
|
|
|
mod_t * get_mod(gchar const * key)
|
|
{
|
|
int i;
|
|
for (i = 0; i < mod_count; i++)
|
|
{
|
|
if (strcmp(mods[i].key, key) == 0)
|
|
return mods + i;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
gboolean append_to_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,
|
|
ICON_COLUMN, generic_mod_icon,
|
|
KEY_COLUMN, mod->key,
|
|
NAME_COLUMN, mod->title,
|
|
-1);
|
|
g_free(key);
|
|
return TRUE;
|
|
}
|
|
g_free(key);
|
|
return FALSE;
|
|
}
|
|
|
|
void mod_metadata_callback(GPid pid, gint status, gpointer data)
|
|
{
|
|
int * out_fd = (int *)data;
|
|
GString * msg = NULL;
|
|
mod_t * mod = mods + mod_count;
|
|
GtkTreeIter iter, mod_iter;
|
|
|
|
mod_count = (mod_count + 1) % MAX_NUM_MODS;
|
|
|
|
free_mod(mod);
|
|
memset(mod, 0, sizeof(mod_t));
|
|
|
|
msg = util_get_output(*out_fd);
|
|
|
|
close(*out_fd);
|
|
free(out_fd);
|
|
|
|
if (!msg)
|
|
return;
|
|
|
|
process_lines(msg, mod_metadata_line, mod);
|
|
|
|
g_string_free(msg, TRUE);
|
|
|
|
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,
|
|
ICON_COLUMN, generic_mod_icon,
|
|
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,
|
|
ICON_COLUMN, generic_mod_icon,
|
|
KEY_COLUMN, mod->key,
|
|
NAME_COLUMN, mod->title,
|
|
-1);
|
|
}
|
|
else
|
|
{
|
|
gtk_tree_model_foreach(GTK_TREE_MODEL(tree_store), append_to_mod, mod);
|
|
}
|
|
}
|
|
|
|
typedef struct tree_node
|
|
{
|
|
gchar const * key;
|
|
gchar * node_path;
|
|
} tree_node;
|
|
|
|
gboolean find_mod(GtkTreeModel * model, GtkTreePath * path,
|
|
GtkTreeIter * iter, gpointer data)
|
|
{
|
|
tree_node * n = (tree_node *)data;
|
|
gchar * key;
|
|
|
|
gtk_tree_model_get(model, iter,
|
|
KEY_COLUMN, &key,
|
|
-1);
|
|
|
|
if (!key)
|
|
return FALSE;
|
|
|
|
if (0 == strcmp(n->key, key))
|
|
{
|
|
n->node_path = gtk_tree_path_to_string(path);
|
|
g_free(key);
|
|
return TRUE;
|
|
}
|
|
|
|
g_free(key);
|
|
return FALSE;
|
|
}
|
|
|
|
void last_mod_callback(GPid pid, gint status, gpointer data)
|
|
{
|
|
int * out_fd = (int *)data;
|
|
gchar * comma_pos = 0, * newline_pos = 0;
|
|
GString * msg = NULL;
|
|
tree_node n;
|
|
|
|
memset(&n, 0, sizeof(tree_node));
|
|
|
|
msg = util_get_output(*out_fd);
|
|
|
|
close(*out_fd);
|
|
free(out_fd);
|
|
|
|
if (!msg)
|
|
return;
|
|
|
|
if (g_str_has_prefix(msg->str, "Error:"))
|
|
{
|
|
g_string_truncate(msg, 2);
|
|
g_string_overwrite(msg, 0, "ra");
|
|
}
|
|
else if (NULL != (comma_pos = g_strstr_len(msg->str, -1, ",")))
|
|
*comma_pos = '\0';
|
|
else if (NULL != (newline_pos = g_strstr_len(msg->str, -1, "\n")))
|
|
*newline_pos = '\0';
|
|
|
|
n.key = msg->str;
|
|
|
|
gtk_tree_model_foreach(GTK_TREE_MODEL(tree_store), find_mod, &n);
|
|
|
|
if (n.node_path)
|
|
{
|
|
GtkTreePath * path;
|
|
path = gtk_tree_path_new_from_string(n.node_path);
|
|
if (path == NULL)
|
|
g_warning("Invalid Path");
|
|
gtk_tree_view_expand_to_path(tree, path);
|
|
gtk_tree_view_set_cursor(tree, path, NULL, FALSE);
|
|
gtk_tree_path_free(path);
|
|
g_free(n.node_path);
|
|
}
|
|
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
void mod_list_line(GString const * mod, gpointer user)
|
|
{
|
|
util_get_mod_metadata(mod->str, mod_metadata_callback);
|
|
}
|
|
|
|
void mod_list_callback(GPid pid, gint status, gpointer data)
|
|
{
|
|
callback_data * d = (callback_data *)data;
|
|
GString * msg = NULL;
|
|
|
|
msg = util_get_output(d->output_fd);
|
|
|
|
close(d->output_fd);
|
|
g_free(d);
|
|
|
|
if (!msg)
|
|
return;
|
|
|
|
mod_count = 0;
|
|
|
|
process_lines(msg, mod_list_line, NULL);
|
|
|
|
util_get_setting("Game.Mods", last_mod_callback);
|
|
|
|
g_string_free(msg, TRUE);
|
|
|
|
g_spawn_close_pid(pid);
|
|
}
|
|
|
|
void tree_view_selection_changed(GtkTreeView * tree_view, gpointer data)
|
|
{
|
|
GtkTreePath * path;
|
|
GtkTreeIter iter;
|
|
gchar * key;
|
|
GString * url;
|
|
|
|
gtk_tree_view_get_cursor(tree_view, &path, NULL);
|
|
|
|
if (path == NULL)
|
|
return;
|
|
|
|
gtk_tree_model_get_iter(GTK_TREE_MODEL(tree_store), &iter, path);
|
|
|
|
gtk_tree_model_get(GTK_TREE_MODEL(tree_store), &iter,
|
|
KEY_COLUMN, &key,
|
|
-1);
|
|
|
|
if (!key)
|
|
{
|
|
gtk_tree_path_free(path);
|
|
return;
|
|
}
|
|
|
|
url = g_string_new(NULL);
|
|
|
|
g_string_printf(url, "http://localhost:%d/mods/%s/mod.html", WEBSERVER_PORT, key);
|
|
|
|
webkit_web_view_load_uri(browser, url->str);
|
|
|
|
g_free(key);
|
|
gtk_tree_path_free(path);
|
|
g_string_free(url, TRUE);
|
|
}
|
|
|
|
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);
|
|
g_signal_connect(tree, "cursor-changed",
|
|
G_CALLBACK(tree_view_selection_changed), 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);
|
|
}
|
|
|
|
void renderer_callback(GPid pid, gint status, gpointer data)
|
|
{
|
|
int * fd = (int *)data;
|
|
GString * msg;
|
|
|
|
msg = util_get_output(*fd);
|
|
|
|
close(*fd);
|
|
g_free(fd);
|
|
|
|
if (!msg)
|
|
return;
|
|
|
|
if (g_str_has_prefix(msg->str, "Error:"))
|
|
{
|
|
g_string_free(msg, TRUE);
|
|
return;
|
|
}
|
|
|
|
if (0 == strcmp(msg->str, "Gl"))
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(gl_button), TRUE);
|
|
else
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(cg_button), TRUE);
|
|
|
|
g_string_free(msg, TRUE);
|
|
}
|
|
|
|
void renderer_changed(GtkToggleButton * widget, gpointer user_data)
|
|
{
|
|
if (!gtk_toggle_button_get_active(widget))
|
|
return;
|
|
|
|
if (GTK_RADIO_BUTTON(widget) == gl_button)
|
|
renderer = RENDERER_GL;
|
|
else
|
|
renderer = RENDERER_CG;
|
|
}
|
|
|
|
int get_renderer(void)
|
|
{
|
|
return renderer;
|
|
}
|
|
|
|
int main(int argc, char ** argv)
|
|
{
|
|
GtkWidget * hbox1, * hbox2, * vbox;
|
|
|
|
int res = chdir("/usr/share/openra");
|
|
|
|
if (0 != res)
|
|
res = chdir("/usr/local/share/openra");
|
|
|
|
if (0 != res)
|
|
{
|
|
g_error("Couldn't change to OpenRA working directory");
|
|
return 1;
|
|
}
|
|
|
|
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);
|
|
|
|
generic_mod_icon = gdk_pixbuf_new_from_file_at_size("soviet-logo.png",
|
|
16, 16, NULL);
|
|
|
|
make_tree_view();
|
|
|
|
util_get_mod_list(mod_list_callback);
|
|
|
|
vbox = gtk_vbox_new(FALSE, 0);
|
|
hbox1 = gtk_hbox_new(FALSE, 0);
|
|
hbox2 = gtk_hbox_new(FALSE, 0);
|
|
|
|
gtk_widget_set_size_request(GTK_WIDGET(tree), 250, 0);
|
|
|
|
gtk_box_pack_end(GTK_BOX(hbox1), GTK_WIDGET(browser), TRUE, TRUE, 0);
|
|
gtk_box_pack_start(GTK_BOX(hbox1), GTK_WIDGET(tree), TRUE, TRUE, 0);
|
|
|
|
gl_button = GTK_RADIO_BUTTON(gtk_radio_button_new_with_label(NULL, "GL Renderer"));
|
|
g_signal_connect(gl_button, "toggled", G_CALLBACK(renderer_changed), 0);
|
|
cg_button = GTK_RADIO_BUTTON(gtk_radio_button_new_with_label_from_widget(gl_button, "CG Renderer"));
|
|
g_signal_connect(cg_button, "toggled", G_CALLBACK(renderer_changed), 0);
|
|
|
|
util_get_setting("Graphics.Renderer", renderer_callback);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox2), GTK_WIDGET(gl_button), FALSE, FALSE, 5);
|
|
gtk_box_pack_start(GTK_BOX(hbox2), GTK_WIDGET(cg_button), FALSE, FALSE, 5);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), hbox1, TRUE, TRUE, 0);
|
|
gtk_box_pack_start(GTK_BOX(vbox), hbox2, FALSE, FALSE, 10);
|
|
|
|
gtk_container_add(GTK_CONTAINER(window), vbox);
|
|
|
|
gtk_widget_show_all(GTK_WIDGET(window));
|
|
g_signal_connect(window, "delete-event", G_CALLBACK(window_delete), 0);
|
|
|
|
gtk_main();
|
|
|
|
return 0;
|
|
}
|