/* Copyright (C) 1999-2007 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 "plugin.h" // toolbar implementation class CFlushReloadSelectedToolbarButton : public IToolbarButton { public: virtual const char* getImage() const { return "model_reload_entity.bmp"; } virtual const char* getText() const { return "Reload"; } virtual const char* getTooltip() const { return "Flush & Reload Selected Model"; } virtual void activate() const { DoFlushReloadSelected(); } virtual EType getType() const { return eButton; } }; CFlushReloadSelectedToolbarButton g_flushreloadselected; unsigned int ToolbarButtonCount(){ return 1; } const IToolbarButton* GetToolbarButton( unsigned int index ){ return &g_flushreloadselected; } // ============================================================================= // Pico utility functions #include "picomodel.h" void PicoPrintFunc( int level, const char *str ){ if ( str == NULL ) { return; } switch ( level ) { case PICO_NORMAL: Sys_Printf( "%s\n", str ); break; case PICO_VERBOSE: Sys_FPrintf( SYS_VRB, "%s\n", str ); break; case PICO_WARNING: Sys_Printf( "WARNING: %s\n", str ); break; case PICO_ERROR: Sys_FPrintf( SYS_VRB, "ERROR: %s\n", str ); break; case PICO_FATAL: Sys_Printf( "ERROR: %s\n", str ); break; } } void PicoLoadFileFunc( char *name, byte **buffer, int *bufSize ){ *bufSize = vfsLoadFile( (const char*) name, (void**) buffer, 0 ); } void PicoFreeFileFunc( void* file ){ vfsFreeFile( file ); } static void initialise(){ PicoInit(); PicoSetMallocFunc( malloc ); PicoSetFreeFunc( free ); PicoSetPrintFunc( PicoPrintFunc ); PicoSetLoadFileFunc( PicoLoadFileFunc ); PicoSetFreeFileFunc( PicoFreeFileFunc ); } static void add_model_apis( CSynapseClient& client ){ const picoModule_t** modules = PicoModuleList( NULL ); while ( *modules != NULL ) { const picoModule_t* module = *modules++; if ( module->canload && module->load ) { for ( unsigned int j = 0; module->defaultExts[j] != NULL; j++ ) client.AddAPI( MODEL_MAJOR, module->defaultExts[j], sizeof( _QERPlugModelTable ) ); } } } static bool model_is_supported( const char* extension ){ const picoModule_t** modules = PicoModuleList( NULL ); while ( *modules != NULL ) { const picoModule_t* module = *modules++; if ( module->canload && module->load ) { for ( unsigned int j = 0; module->defaultExts[j] != NULL; j++ ) if ( strcmp( extension, module->defaultExts[j] ) == 0 ) { return true; } } } return false; } void init_filetypes(){ const picoModule_t **modules = PicoModuleList( NULL ); while ( *modules != NULL ) { const picoModule_t* module = *modules++; if ( module->canload && module->load ) { for ( char*const* ext = module->defaultExts; *ext != NULL; ++ext ) { char buf[16]; buf[0] = '*'; buf[1] = '.'; strcpy( buf + 2, *ext ); GetFileTypeRegistry()->addType( MODEL_MAJOR, filetype_t( module->displayName, buf ) ); } } } } // plugin implementation static const char *PLUGIN_NAME = "Model loading module"; static const char *PLUGIN_COMMANDS = "About;-;Flush & Reload Models;Flush & Reload Selected"; static const char *PLUGIN_ABOUT = "Model Module v1.0 for GtkRadiant\nby Arnout van Meer (rr2do2@splashdamage.com)\n\nBased on the MD3Model Module by SPoG\nPicoModel Library Copyright (c) 2002, Randy Reddig & seaw0lf\n\nSupported models:\n"; extern "C" const char* QERPlug_Init( void *hApp, void* pMainWidget ){ init_filetypes(); return (char *) PLUGIN_NAME; } extern "C" const char* QERPlug_GetName(){ return (char *) PLUGIN_NAME; } extern "C" const char* QERPlug_GetCommandList(){ return (char *) PLUGIN_COMMANDS; } extern "C" void QERPlug_Dispatch( const char *p, vec3_t vMin, vec3_t vMax, bool bSingleBrush ){ if ( !strcmp( p, "Flush & Reload Selected" ) ) { DoFlushReloadSelected(); } else if ( !strcmp( p, "Flush & Reload Models" ) ) { DoFlushReloadAll(); } else if ( !strcmp( p, "About" ) ) { const picoModule_t** modules = PicoModuleList( NULL ); char about_buf[1024]; strncpy( about_buf, PLUGIN_ABOUT, sizeof( about_buf ) - 1 ); while ( *modules != NULL ) { const picoModule_t* module = *modules++; strncat( about_buf, module->displayName, sizeof( about_buf ) - 1 ); strncat( about_buf, " (", sizeof( about_buf ) - 1 ); strncat( about_buf, module->defaultExts[0], sizeof( about_buf ) - 1 ); strncat( about_buf, ")\n\t", sizeof( about_buf ) - 1 ); strncat( about_buf, module->copyright, sizeof( about_buf ) - 1 ); strncat( about_buf, "\n", sizeof( about_buf ) - 1 ); } g_FuncTable.m_pfnMessageBox( NULL, about_buf, "About", MB_OK, NULL ); } } void DoFlushReloadSelected() { } void DoFlushReloadAll() { GetModelCache()->RefreshAll(); } // ============================================================================= // function tables _QERFuncTable_1 g_FuncTable; _QERQglTable g_QglTable; _QERShadersTable g_ShadersTable; _QERFileSystemTable g_FileSystemTable; // ============================================================================= // SYNAPSE CSynapseServer* g_pSynapseServer = NULL; CSynapseClientModel g_SynapseClient; static const XMLConfigEntry_t entries[] = { { SHADERS_MAJOR, SYN_REQUIRE, sizeof( g_ShadersTable ), &g_ShadersTable }, { VFS_MAJOR, SYN_REQUIRE, sizeof( g_FileSystemTable ), &g_FileSystemTable }, { NULL, SYN_UNKNOWN, 0, NULL } }; #if __GNUC__ >= 4 #pragma GCC visibility push(default) #endif extern "C" CSynapseClient * SYNAPSE_DLL_EXPORT Synapse_EnumerateInterfaces( const char *version, CSynapseServer *pServer ) { #if __GNUC__ >= 4 #pragma GCC visibility pop #endif if ( strcmp( version, SYNAPSE_VERSION ) ) { Syn_Printf( "ERROR: synapse API version mismatch: should be '" SYNAPSE_VERSION "', got '%s'\n", version ); return NULL; } g_pSynapseServer = pServer; g_pSynapseServer->IncRef(); Set_Syn_Printf( g_pSynapseServer->Get_Syn_Printf() ); initialise(); add_model_apis( g_SynapseClient ); g_SynapseClient.AddAPI( TOOLBAR_MAJOR, "model", sizeof( _QERPlugToolbarTable ) ); g_SynapseClient.AddAPI( PLUGIN_MAJOR, "model", sizeof( _QERPluginTable ) ); g_SynapseClient.AddAPI( RADIANT_MAJOR, NULL, sizeof( g_FuncTable ), SYN_REQUIRE, &g_FuncTable ); g_SynapseClient.AddAPI( QGL_MAJOR, NULL, sizeof( g_QglTable ), SYN_REQUIRE, &g_QglTable ); if ( !g_SynapseClient.ConfigXML( pServer, NULL, entries ) ) { return NULL; } return &g_SynapseClient; } bool CSynapseClientModel::RequestAPI( APIDescriptor_t *pAPI ){ if ( !strcmp( pAPI->major_name, MODEL_MAJOR ) ) { _QERPlugModelTable* pTable = static_cast<_QERPlugModelTable*>( pAPI->mpTable ); if ( model_is_supported( pAPI->minor_name ) ) { pTable->m_pfnLoadModel = &LoadModel; return true; } } else if ( !strcmp( pAPI->major_name, TOOLBAR_MAJOR ) ) { _QERPlugToolbarTable* pTable = static_cast<_QERPlugToolbarTable*>( pAPI->mpTable ); pTable->m_pfnToolbarButtonCount = &ToolbarButtonCount; pTable->m_pfnGetToolbarButton = &GetToolbarButton; return true; } else if ( !strcmp( pAPI->major_name, PLUGIN_MAJOR ) ) { _QERPluginTable* pTable = static_cast<_QERPluginTable*>( pAPI->mpTable ); pTable->m_pfnQERPlug_Init = QERPlug_Init; pTable->m_pfnQERPlug_GetName = QERPlug_GetName; pTable->m_pfnQERPlug_GetCommandList = QERPlug_GetCommandList; pTable->m_pfnQERPlug_Dispatch = QERPlug_Dispatch; return true; } Syn_Printf( "ERROR: RequestAPI( '%s' ) not found in '%s'\n", pAPI->major_name, GetInfo() ); return false; } #include "version.h" const char* CSynapseClientModel::GetInfo(){ return "picomodel loader module built " __DATE__ " " RADIANT_VERSION; } const char* CSynapseClientModel::GetName(){ return "model"; }