Utility app helper functions for listing mods in the GTK+ launcher.

This commit is contained in:
Matthew Bowra-Dean
2010-11-30 23:17:30 +13:00
committed by Chris Forbes
parent 6fbc7bc7c4
commit 650195e31c
3 changed files with 233 additions and 0 deletions

View File

@@ -13,11 +13,13 @@
#include "server.h"
#include "bridge.h"
#include "utility.h"
#define WEBSERVER_PORT 48764
GtkWindow * window;
WebKitWebView * browser;
GtkTreeStore * treeStore;
gboolean window_delete(GtkWidget * widget, GdkEvent * event,
gpointer user_data)
@@ -27,6 +29,140 @@ gboolean window_delete(GtkWidget * widget, GdkEvent * event,
return FALSE;
}
enum
{
ICON_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)
{
char * c;
c = strtok(lines, "\n");
while (c != NULL)
{
cb(c, data);
c = strtok(NULL, "\n");
}
}
#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;
g_message("Mod standalone: %d", mod->standalone);
}
}
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];
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);
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);
}
int main(int argc, char ** argv)
{
server_init(WEBSERVER_PORT);
@@ -41,6 +177,10 @@ int main(int argc, char ** argv)
g_signal_connect(browser, "window-object-cleared",
G_CALLBACK(bind_js_bridge), 0);
treeStore = gtk_tree_store_new(N_COLUMNS, GDK_TYPE_PIXBUF, G_TYPE_STRING);
util_get_mod_list(mod_list_callback);
gtk_container_add(GTK_CONTAINER(window), GTK_WIDGET(browser));
//TODO: Load the mod html file based on selected mod in launcher

View File

@@ -0,0 +1,82 @@
/*
* 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 <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <glib.h>
int util_get_mod_list (GChildWatchFunc callback)
{
GPid child_pid;
gint * out_fd = (gint *)malloc(sizeof(gint));
char * spawn_args[] = { "mono", "OpenRA.Utility.exe", "-l", NULL };
gboolean result = g_spawn_async_with_pipes(NULL, spawn_args, NULL,
G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD,
NULL, NULL, &child_pid, NULL, out_fd, NULL, NULL);
if (!result)
{
return FALSE;
}
g_child_watch_add(child_pid, callback, out_fd);
return TRUE;
}
int util_get_mod_metadata(char const * mod, GChildWatchFunc callback)
{
GPid child_pid;
gint * out_fd = (gint *)malloc(sizeof(gint));
char * spawn_args[] = { "mono", "OpenRA.Utility.exe", NULL, NULL };
char util_args[32];
gboolean result;
sprintf(util_args, "-i=%s", mod);
spawn_args[2] = util_args;
result = g_spawn_async_with_pipes(NULL, spawn_args, NULL,
G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD,
NULL, NULL, &child_pid, NULL, out_fd, NULL, NULL);
if (!result)
{
return FALSE;
}
g_child_watch_add(child_pid, callback, out_fd);
return TRUE;
}
char * util_get_output(int fd, int * output_len)
{
char buffer[1024], * msg = NULL;
int read_bytes = 0;
*output_len = 0;
while (0 != (read_bytes = read(fd, buffer, 1024)))
{
if (-1 == read_bytes)
{
g_error("Error reading from command output");
free(msg);
*output_len = 0;
return NULL;
}
*output_len += read_bytes;
msg = (char *)realloc(msg, *output_len + 1);
memcpy(msg + (*output_len - read_bytes), buffer, read_bytes);
}
msg[*output_len] = '\0';
return msg;
}

View File

@@ -0,0 +1,11 @@
/*
* 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.
*/
int util_get_mod_list (GChildWatchFunc);
int util_get_mod_metadata(char const *, GChildWatchFunc);
char * util_get_output(int, int *);