3 Copyright (C) 1999-2007 id Software, Inc. and contributors.
\r
4 For a list of contributors, see the accompanying CONTRIBUTORS file.
\r
6 This file is part of GtkRadiant.
\r
8 GtkRadiant is free software; you can redistribute it and/or modify
\r
9 it under the terms of the GNU General Public License as published by
\r
10 the Free Software Foundation; either version 2 of the License, or
\r
11 (at your option) any later version.
\r
13 GtkRadiant is distributed in the hope that it will be useful,
\r
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
\r
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
\r
16 GNU General Public License for more details.
\r
18 You should have received a copy of the GNU General Public License
\r
19 along with GtkRadiant; if not, write to the Free Software
\r
20 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
\r
25 #if 0 // stop using windowing systems in plugins - put the text in SynapseClient::GetInfo
\r
26 // =============================================================================
\r
27 // Utility functions
\r
28 static void dialog_button_callback (GtkWidget *widget, gpointer data)
\r
33 parent = gtk_widget_get_toplevel (widget);
\r
34 loop = (int*)g_object_get_data (G_OBJECT (parent), "loop");
\r
35 ret = (int*)g_object_get_data (G_OBJECT (parent), "ret");
\r
41 static gint dialog_delete_callback (GtkWidget *widget, GdkEvent* event, gpointer data)
\r
45 gtk_widget_hide (widget);
\r
46 loop = (int*)g_object_get_data (G_OBJECT (widget), "loop");
\r
52 int DoAboutBox( GtkWidget *parent )
\r
54 GtkWidget *window, *w, *text, *vbox, *hbox, *hbox2, *frame;
\r
60 const picoModule_t **modules, *pm;
\r
62 window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
\r
63 gtk_signal_connect (GTK_OBJECT (window), "delete_event",
\r
64 GTK_SIGNAL_FUNC (dialog_delete_callback), NULL);
\r
65 gtk_signal_connect (GTK_OBJECT (window), "destroy",
\r
66 GTK_SIGNAL_FUNC (gtk_widget_destroy), NULL);
\r
67 gtk_window_set_title (GTK_WINDOW (window), "About...");
\r
68 gtk_container_border_width (GTK_CONTAINER (window), 10);
\r
69 g_object_set_data (G_OBJECT (window), "loop", &loop);
\r
70 g_object_set_data (G_OBJECT (window), "ret", &ret);
\r
71 gtk_widget_realize (window);
\r
74 gtk_window_set_transient_for (GTK_WINDOW (window), GTK_WINDOW (parent));
\r
76 vbox = gtk_vbox_new (FALSE, 10);
\r
77 gtk_container_add (GTK_CONTAINER (window), vbox);
\r
78 gtk_widget_show (vbox);
\r
80 style = gtk_widget_get_style(window);
\r
82 hbox2 = gtk_hbox_new (FALSE, 10);
\r
83 gtk_box_pack_start (GTK_BOX (vbox), hbox2, FALSE, FALSE, 2);
\r
84 gtk_widget_show (hbox2);
\r
86 frame = gtk_frame_new (NULL);
\r
87 gtk_box_pack_start (GTK_BOX (hbox2), frame, FALSE, FALSE, 2);
\r
88 gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
\r
89 gtk_widget_show (frame);
\r
91 if( g_FuncTable.m_pfnLoadBitmap( "picomodel.bmp", (void **)&pixmap, (void **)&mask ) ) {
\r
92 w = gtk_pixmap_new (pixmap, mask);
\r
93 gtk_container_add (GTK_CONTAINER (frame), w);
\r
94 gtk_widget_show (w);
\r
97 w = gtk_label_new ("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" );
\r
98 gtk_box_pack_start (GTK_BOX (vbox), w, FALSE, FALSE, 2);
\r
99 gtk_label_set_justify (GTK_LABEL (w), GTK_JUSTIFY_LEFT);
\r
100 gtk_widget_show (w);
\r
102 w = gtk_scrolled_window_new(NULL, NULL);
\r
103 gtk_box_pack_start(GTK_BOX(vbox), w, TRUE, TRUE, 2);
\r
104 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(w), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
\r
105 gtk_widget_show(w);
\r
107 text = gtk_text_new(NULL, NULL);
\r
108 gtk_text_set_editable(GTK_TEXT(text), FALSE);
\r
109 gtk_container_add(GTK_CONTAINER(w), text);
\r
111 strcpy( buf, "#Supported Model Formats:\n" );
\r
112 gtk_text_insert(GTK_TEXT(text), NULL, NULL, NULL, buf, -1);
\r
114 for( modules = PicoModuleList( NULL ); *modules != NULL; modules++ )
\r
121 sprintf( buf, "\n%s, version %s, (c) %s", pm->displayName, pm->version, pm->copyright );
\r
122 gtk_text_insert(GTK_TEXT(text), NULL, NULL, NULL, buf, -1);
\r
125 gtk_text_set_word_wrap(GTK_TEXT(text), FALSE);
\r
126 gtk_widget_show(text);
\r
128 gtk_text_set_point(GTK_TEXT(text), 0);
\r
129 gtk_text_forward_delete(GTK_TEXT(text), 1);
\r
131 w = gtk_hseparator_new ();
\r
132 gtk_box_pack_start (GTK_BOX (vbox), w, FALSE, FALSE, 2);
\r
133 gtk_widget_show (w);
\r
135 hbox = gtk_hbox_new (FALSE, 10);
\r
136 gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 2);
\r
137 gtk_widget_show (hbox);
\r
139 w = gtk_button_new_with_label ("Ok");
\r
140 gtk_box_pack_start (GTK_BOX (hbox), w, TRUE, TRUE, 0);
\r
141 gtk_signal_connect (GTK_OBJECT (w), "clicked",
\r
142 GTK_SIGNAL_FUNC (dialog_button_callback), GINT_TO_POINTER (IDOK));
\r
143 GTK_WIDGET_SET_FLAGS (w, GTK_CAN_DEFAULT);
\r
144 gtk_widget_grab_default (w);
\r
145 gtk_widget_show (w);
\r
148 gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
\r
149 gtk_widget_show (window);
\r
150 gtk_grab_add (window);
\r
153 gtk_main_iteration ();
\r
155 gtk_grab_remove (window);
\r
156 gtk_widget_destroy (window);
\r
162 // toolbar implementation
\r
164 class CFlushReloadSelectedToolbarButton : public IToolbarButton
\r
167 virtual const char* getImage() const
\r
169 return "model_reload_entity.bmp";
\r
171 virtual const char* getText() const
\r
175 virtual const char* getTooltip() const
\r
177 return "Flush & Reload Selected Model";
\r
179 virtual void activate() const
\r
181 DoFlushReloadSelected();
\r
183 virtual EType getType() const
\r
189 CFlushReloadSelectedToolbarButton g_flushreloadselected;
\r
191 unsigned int ToolbarButtonCount()
\r
196 const IToolbarButton* GetToolbarButton(unsigned int index)
\r
198 return &g_flushreloadselected;
\r
201 // =============================================================================
\r
202 // Pico utility functions
\r
204 #include "picomodel.h"
\r
206 void PicoPrintFunc( int level, const char *str )
\r
213 Sys_Printf( "%s\n", str );
\r
217 Sys_FPrintf( SYS_VRB, "%s\n", str );
\r
221 Sys_Printf( "WARNING: %s\n", str );
\r
225 Sys_FPrintf( SYS_VRB, "ERROR: %s\n", str );
\r
229 Sys_Printf( "ERROR: %s\n", str );
\r
234 void PicoLoadFileFunc( char *name, byte **buffer, int *bufSize )
\r
236 *bufSize = vfsLoadFile( (const char*) name, (void**) buffer, 0 );
\r
239 void PicoFreeFileFunc( void* file )
\r
244 static void initialise()
\r
247 PicoSetMallocFunc( malloc );
\r
248 PicoSetFreeFunc( free );
\r
249 PicoSetPrintFunc( PicoPrintFunc );
\r
250 PicoSetLoadFileFunc( PicoLoadFileFunc );
\r
251 PicoSetFreeFileFunc( PicoFreeFileFunc );
\r
254 static void add_model_apis(CSynapseClient& client)
\r
256 const picoModule_t** modules = PicoModuleList( NULL );
\r
257 while(*modules != NULL)
\r
259 const picoModule_t* module = *modules++;
\r
260 if(module->canload && module->load)
\r
261 for(unsigned int j = 0; module->defaultExts[j] != NULL; j++)
\r
262 client.AddAPI(MODEL_MAJOR, module->defaultExts[j], sizeof(_QERPlugModelTable));
\r
266 static bool model_is_supported(const char* extension)
\r
268 const picoModule_t** modules = PicoModuleList( NULL );
\r
269 while(*modules != NULL)
\r
271 const picoModule_t* module = *modules++;
\r
272 if(module->canload && module->load)
\r
273 for(unsigned int j = 0; module->defaultExts[j] != NULL; j++)
\r
274 if(strcmp(extension, module->defaultExts[j]) == 0)
\r
280 void init_filetypes()
\r
282 const picoModule_t **modules = PicoModuleList(NULL);
\r
283 while(*modules != NULL)
\r
285 const picoModule_t* module = *modules++;
\r
286 if(module->canload && module->load)
\r
288 for(char*const* ext = module->defaultExts; *ext != NULL; ++ext)
\r
293 strcpy(buf+2, *ext);
\r
294 GetFileTypeRegistry()->addType(MODEL_MAJOR, filetype_t(module->displayName, buf));
\r
300 // plugin implementation
\r
302 static const char *PLUGIN_NAME = "Model loading module";
\r
303 static const char *PLUGIN_COMMANDS = "Flush & Reload Models,Flush & Reload Selected";
\r
304 static const char *PLUGIN_ABOUT = "Model loading module";
\r
306 extern "C" const char* QERPlug_Init (void *hApp, void* pMainWidget)
\r
309 return (char *) PLUGIN_NAME;
\r
312 extern "C" const char* QERPlug_GetName ()
\r
314 return (char *) PLUGIN_NAME;
\r
317 extern "C" const char* QERPlug_GetCommandList ()
\r
319 return (char *) PLUGIN_COMMANDS;
\r
322 extern "C" void QERPlug_Dispatch (const char *p, vec3_t vMin, vec3_t vMax, bool bSingleBrush)
\r
324 if( !strcmp( p, "Flush & Reload Selected" ) )
\r
325 DoFlushReloadSelected();
\r
326 else if( !strcmp( p, "Flush & Reload Models" ) )
\r
327 DoFlushReloadAll();
\r
331 void DoFlushReloadSelected() {
\r
334 void DoFlushReloadAll() {
\r
335 GetModelCache()->RefreshAll();
\r
338 // =============================================================================
\r
341 _QERFuncTable_1 g_FuncTable;
\r
342 _QERQglTable g_QglTable;
\r
343 _QERShadersTable g_ShadersTable;
\r
344 _QERFileSystemTable g_FileSystemTable;
\r
346 // =============================================================================
\r
349 CSynapseServer* g_pSynapseServer = NULL;
\r
350 CSynapseClientModel g_SynapseClient;
\r
352 static const XMLConfigEntry_t entries[] =
\r
353 { { SHADERS_MAJOR, SYN_REQUIRE, sizeof(g_ShadersTable), &g_ShadersTable },
\r
354 { VFS_MAJOR, SYN_REQUIRE, sizeof(g_FileSystemTable), &g_FileSystemTable },
\r
355 { NULL, SYN_UNKNOWN, 0, NULL } };
\r
358 #pragma GCC visibility push(default)
\r
360 extern "C" CSynapseClient* SYNAPSE_DLL_EXPORT Synapse_EnumerateInterfaces( const char *version, CSynapseServer *pServer ) {
\r
362 #pragma GCC visibility pop
\r
364 if (strcmp(version, SYNAPSE_VERSION))
\r
366 Syn_Printf("ERROR: synapse API version mismatch: should be '" SYNAPSE_VERSION "', got '%s'\n", version);
\r
369 g_pSynapseServer = pServer;
\r
370 g_pSynapseServer->IncRef();
\r
371 Set_Syn_Printf( g_pSynapseServer->Get_Syn_Printf() );
\r
375 add_model_apis(g_SynapseClient);
\r
376 g_SynapseClient.AddAPI(TOOLBAR_MAJOR, "model", sizeof(_QERPlugToolbarTable));
\r
377 g_SynapseClient.AddAPI(PLUGIN_MAJOR, "model", sizeof(_QERPluginTable));
\r
379 g_SynapseClient.AddAPI(RADIANT_MAJOR, NULL, sizeof(g_FuncTable), SYN_REQUIRE, &g_FuncTable);
\r
380 g_SynapseClient.AddAPI(QGL_MAJOR, NULL, sizeof(g_QglTable), SYN_REQUIRE, &g_QglTable);
\r
382 if ( !g_SynapseClient.ConfigXML( pServer, NULL, entries ) ) {
\r
386 return &g_SynapseClient;
\r
389 bool CSynapseClientModel::RequestAPI(APIDescriptor_t *pAPI)
\r
391 if (!strcmp(pAPI->major_name, MODEL_MAJOR))
\r
393 _QERPlugModelTable* pTable= static_cast<_QERPlugModelTable*>(pAPI->mpTable);
\r
395 if (model_is_supported(pAPI->minor_name))
\r
397 pTable->m_pfnLoadModel = &LoadModel;
\r
401 else if (!strcmp(pAPI->major_name, TOOLBAR_MAJOR))
\r
403 _QERPlugToolbarTable* pTable= static_cast<_QERPlugToolbarTable*>(pAPI->mpTable);
\r
405 pTable->m_pfnToolbarButtonCount = &ToolbarButtonCount;
\r
406 pTable->m_pfnGetToolbarButton = &GetToolbarButton;
\r
409 else if (!strcmp(pAPI->major_name, PLUGIN_MAJOR))
\r
411 _QERPluginTable* pTable= static_cast<_QERPluginTable*>(pAPI->mpTable);
\r
413 pTable->m_pfnQERPlug_Init = QERPlug_Init;
\r
414 pTable->m_pfnQERPlug_GetName = QERPlug_GetName;
\r
415 pTable->m_pfnQERPlug_GetCommandList = QERPlug_GetCommandList;
\r
416 pTable->m_pfnQERPlug_Dispatch = QERPlug_Dispatch;
\r
420 Syn_Printf("ERROR: RequestAPI( '%s' ) not found in '%s'\n", pAPI->major_name, GetInfo());
\r
424 #include "version.h"
\r
426 const char* CSynapseClientModel::GetInfo()
\r
428 return "picomodel loader module built " __DATE__ " " RADIANT_VERSION;
\r
431 const char* CSynapseClientModel::GetName()
\r