+#ifdef SUPPORT_GECKO\r
+\r
+// includes everything!\r
+#include <OffscreenGecko/browser.h>\r
+\r
+#ifdef _MSC_VER\r
+# pragma comment( lib, "OffscreenGecko" )\r
+#endif\r
+\r
+#include "quakedef.h"\r
+#include "cl_dyntexture.h"\r
+#include "cl_gecko.h"\r
+\r
+static rtexturepool_t *cl_geckotexturepool;\r
+static OSGK_Embedding *cl_geckoembedding;\r
+\r
+struct clgecko_s {\r
+ qboolean active;\r
+ char name[ MAX_QPATH + 32 ];\r
+\r
+ OSGK_Browser *browser;\r
+ \r
+ rtexture_t *texture;\r
+};\r
+\r
+static clgecko_t cl_geckoinstances[ MAX_GECKO_INSTANCES ];\r
+\r
+static clgecko_t * cl_gecko_findunusedinstance( void ) {\r
+ int i;\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( !instance->active ) {\r
+ return instance;\r
+ }\r
+ }\r
+ return NULL;\r
+}\r
+\r
+clgecko_t * CL_Gecko_FindBrowser( const char *name ) {\r
+ int i;\r
+\r
+ if( !name || !*name || strncmp( name, CLGECKOPREFIX, sizeof( CLGECKOPREFIX ) - 1 ) != 0 ) {\r
+ if( developer.integer > 0 ) {\r
+ Con_Printf( "CL_Gecko_FindBrowser: Bad gecko texture name '%s'!\n", name );\r
+ }\r
+ return NULL;\r
+ }\r
+\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( instance->active && strcmp( instance->name, name ) == 0 ) {\r
+ return instance;\r
+ }\r
+ }\r
+\r
+ return NULL;\r
+}\r
+\r
+static void cl_gecko_updatecallback( rtexture_t *texture, clgecko_t *instance ) {\r
+ const unsigned char *data;\r
+ if( instance->browser ) {\r
+ // TODO: OSGK only supports BGRA right now\r
+ data = osgk_browser_lock_data( instance->browser, NULL );\r
+ R_UpdateTexture( texture, data, 0, 0, DEFAULT_GECKO_WIDTH, DEFAULT_GECKO_HEIGHT );\r
+ osgk_browser_unlock_data( instance->browser, data );\r
+ }\r
+}\r
+\r
+static void cl_gecko_linktexture( clgecko_t *instance ) {\r
+ // TODO: assert that instance->texture == NULL\r
+ instance->texture = R_LoadTexture2D( cl_geckotexturepool, instance->name, DEFAULT_GECKO_WIDTH, DEFAULT_GECKO_HEIGHT, NULL, TEXTYPE_RGBA, TEXF_ALPHA, NULL );\r
+ R_MakeTextureDynamic( instance->texture, cl_gecko_updatecallback, instance );\r
+ CL_LinkDynTexture( instance->name, instance->texture );\r
+}\r
+\r
+static void cl_gecko_unlinktexture( clgecko_t *instance ) {\r
+ if( instance->texture ) {\r
+ CL_UnlinkDynTexture( instance->name );\r
+ R_FreeTexture( instance->texture );\r
+ instance->texture = NULL;\r
+ }\r
+}\r
+\r
+clgecko_t * CL_Gecko_CreateBrowser( const char *name ) {\r
+ // TODO: verify that we dont use a name twice\r
+ clgecko_t *instance = cl_gecko_findunusedinstance();\r
+ // TODO: assert != NULL\r
+ \r
+ instance->active = true;\r
+ strlcpy( instance->name, name, sizeof( instance->name ) );\r
+ instance->browser = osgk_browser_create( cl_geckoembedding, DEFAULT_GECKO_WIDTH, DEFAULT_GECKO_HEIGHT );\r
+ // TODO: assert != NULL\r
+\r
+ cl_gecko_linktexture( instance );\r
+\r
+ return instance;\r
+}\r
+\r
+void CL_Gecko_DestroyBrowser( clgecko_t *instance ) {\r
+ if( !instance || !instance->active ) {\r
+ return;\r
+ }\r
+\r
+ instance->active = false;\r
+ cl_gecko_unlinktexture( instance );\r
+\r
+ osgk_release( instance->browser );\r
+ instance->browser = NULL;\r
+}\r
+\r
+void CL_Gecko_Frame( void ) {\r
+ int i;\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( instance->active ) {\r
+ if( instance->browser && osgk_browser_query_dirty( instance->browser ) == 1 ) {\r
+ R_MarkDirtyTexture( instance->texture );\r
+ }\r
+ }\r
+ }\r
+}\r
+\r
+static void cl_gecko_start( void )\r
+{\r
+ int i;\r
+ cl_geckotexturepool = R_AllocTexturePool();\r
+\r
+ // recreate textures on module start\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( instance->active ) {\r
+ cl_gecko_linktexture( instance );\r
+ }\r
+ }\r
+}\r
+\r
+static void cl_gecko_shutdown( void )\r
+{\r
+ int i;\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( instance->active ) {\r
+ cl_gecko_unlinktexture( instance );\r
+ }\r
+ }\r
+ R_FreeTexturePool( &cl_geckotexturepool );\r
+}\r
+\r
+static void cl_gecko_newmap( void )\r
+{\r
+ // DO NOTHING\r
+}\r
+\r
+void CL_Gecko_Shutdown( void ) {\r
+ int i;\r
+ for( i = 0 ; i < MAX_GECKO_INSTANCES ; i++ ) {\r
+ clgecko_t *instance = &cl_geckoinstances[ i ];\r
+ if( instance->active ) {\r
+ cl_gecko_unlinktexture( instance );\r
+ } \r
+ }\r
+ osgk_release( cl_geckoembedding );\r
+}\r
+\r
+static void cl_gecko_create_f( void ) {\r
+ char name[MAX_QPATH];\r
+\r
+ if (Cmd_Argc() != 2)\r
+ {\r
+ Con_Print("usage: gecko_create <name>\npcreates a browser (full texture path " CLGECKOPREFIX "<name>)\n");\r
+ return;\r
+ }\r
+\r
+ // TODO: use snprintf instead\r
+ sprintf(name, CLGECKOPREFIX "%s", Cmd_Argv(1));\r
+ CL_Gecko_CreateBrowser( name );\r
+}\r
+\r
+static void cl_gecko_destroy_f( void ) {\r
+ char name[MAX_QPATH];\r
+\r
+ if (Cmd_Argc() != 2)\r
+ {\r
+ Con_Print("usage: gecko_destroy <name>\ndestroys a browser (full texture path " CLGECKOPREFIX "<name>)\n");\r
+ return;\r
+ }\r
+\r
+ // TODO: use snprintf instead\r
+ sprintf(name, CLGECKOPREFIX "%s", Cmd_Argv(1));\r
+ CL_Gecko_DestroyBrowser( CL_Gecko_FindBrowser( name ) );\r
+}\r
+\r
+static void cl_gecko_navigate_f( void ) {\r
+ char name[MAX_QPATH];\r
+ const char *URI;\r
+\r
+ if (Cmd_Argc() != 3)\r
+ {\r
+ Con_Print("usage: gecko_destroy <name> <URI>\nnavigates to a certain URI (full texture path " CLGECKOPREFIX "<name>)\n");\r
+ return;\r
+ }\r
+\r
+ // TODO: use snprintf instead\r
+ sprintf(name, CLGECKOPREFIX "%s", Cmd_Argv(1));\r
+ URI = Cmd_Argv( 2 );\r
+ CL_Gecko_NavigateToURI( CL_Gecko_FindBrowser( name ), URI );\r
+}\r
+\r
+void CL_Gecko_Init( void )\r
+{\r
+ OSGK_EmbeddingOptions *options = osgk_embedding_options_create();\r
+ osgk_embedding_options_add_search_path( options, "./xulrunner/" );\r
+ cl_geckoembedding = osgk_embedding_create_with_options( options, NULL );\r
+ osgk_release( options );\r
+ \r
+ if( cl_geckoembedding == NULL ) {\r
+ Con_Printf( "CL_Gecko_Init: Couldn't retrieve gecko embedding object!\n" );\r
+ }\r
+ \r
+ Cmd_AddCommand( "gecko_create", cl_gecko_create_f, "Create a gecko browser instance" );\r
+ Cmd_AddCommand( "gecko_destroy", cl_gecko_destroy_f, "Destroy a gecko browser instance" );\r
+ Cmd_AddCommand( "gecko_navigate", cl_gecko_navigate_f, "Navigate a gecko browser to an URI" );\r
+\r
+ R_RegisterModule( "CL_Gecko", cl_gecko_start, cl_gecko_shutdown, cl_gecko_newmap );\r
+}\r
+\r
+void CL_Gecko_NavigateToURI( clgecko_t *instance, const char *URI ) {\r
+ if( instance && instance->active ) {\r
+ osgk_browser_navigate( instance->browser, URI );\r
+ }\r
+}\r
+\r
+// TODO: write this function\r
+void CL_Gecko_Event_CursorMove( clgecko_t *instance, float x, float y );\r
+qboolean CL_Gecko_Event_Key( clgecko_t *instance, int key, clgecko_buttoneventtype_t eventtype );\r
+\r
+#endif
\ No newline at end of file