/* Copyright (C) 1999-2006 Id Software, Inc. and contributors. For a list of contributors, see the accompanying CONTRIBUTORS file. This file is part of GtkRadiant. GtkRadiant 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. GtkRadiant 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 GtkRadiant; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "help.h" #include "debugging/debugging.h" #include #include #include "libxml/parser.h" #include "generic/callback.h" #include "gtkutil/menu.h" #include "stream/stringstream.h" #include "os/file.h" #include "url.h" #include "preferences.h" #include "mainframe.h" /*! the urls to fire up in the game packs help menus */ namespace { std::list mHelpURLs; } /*! needed for hooking in Gtk+ */ void HandleHelpCommand( CopiedString& str ){ OpenURL( str.c_str() ); } void process_xlink( const char* filename, const char *menu_name, const char *base_url, GtkMenu *menu ){ if ( file_exists( filename ) ) { xmlDocPtr pDoc = xmlParseFile( filename ); if ( pDoc ) { globalOutputStream() << "Processing .xlink file '" << filename << "'\n"; // create sub menu GtkMenu* menu_in_menu = create_sub_menu_with_mnemonic( menu, menu_name ); if ( g_Layout_enableDetachableMenus.m_value ) { menu_tearoff( menu_in_menu ); } // start walking the nodes, find the 'links' one xmlNodePtr pNode = pDoc->children; while ( pNode && strcmp( (const char*)pNode->name, "links" ) ) pNode = pNode->next; if ( pNode ) { pNode = pNode->children; while ( pNode ) { if ( !strcmp( (const char*)pNode->name, "item" ) ) { // process the URL CopiedString url; xmlChar* prop = xmlGetProp( pNode, reinterpret_cast( "url" ) ); ASSERT_NOTNULL( prop ); if ( strstr( reinterpret_cast( prop ), "http://" ) ) { // complete URL url = reinterpret_cast( prop ); } else { // relative URL StringOutputStream full( 256 ); full << base_url << reinterpret_cast( prop ); url = full.c_str(); } mHelpURLs.push_back( url ); xmlFree( prop ); prop = xmlGetProp( pNode, reinterpret_cast( "name" ) ); ASSERT_NOTNULL( prop ); create_menu_item_with_mnemonic( menu_in_menu, reinterpret_cast( prop ), ReferenceCaller( mHelpURLs.back() ) ); xmlFree( prop ); } pNode = pNode->next; } } xmlFreeDoc( pDoc ); } else { globalOutputStream() << "'" << filename << "' parse failed\n"; } } else { globalOutputStream() << "'" << filename << "' not found\n"; } } void create_game_help_menu( GtkMenu *menu ){ StringOutputStream filename( 256 ); filename << AppPath_get() << "global.xlink"; process_xlink( filename.c_str(), "General", AppPath_get(), menu ); #if 1 filename.clear(); filename << g_pGameDescription->mGameToolsPath.c_str() << "game.xlink"; process_xlink( filename.c_str(), g_pGameDescription->getRequiredKeyValue( "name" ), g_pGameDescription->mGameToolsPath.c_str(), menu ); #else for ( std::list::iterator iGame = g_GamesDialog.mGames.begin(); iGame != g_GamesDialog.mGames.end(); ++iGame ) { filename.clear(); filename << ( *iGame )->mGameToolsPath.c_str() << "game.xlink"; process_xlink( filename.c_str(), ( *iGame )->getRequiredKeyValue( "name" ), ( *iGame )->mGameToolsPath.c_str(), menu ); } #endif }