/*
-Copyright (C) 1999-2006 Id Software, Inc. and contributors.
+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.
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
-#include "eclass_def.h"
+#include "cmdlib.h"
+#include "synapse.h"
+#define USE_QERTABLE_DEFINE
+#include "qerplugin.h"
+#define USE_ECLASSMANAGER_DEFINE
+#include "ieclass.h"
+#define USE_SCRIPLIBTABLE_DEFINE
#include "iscriplib.h"
+
+#define __VFSTABLENAME g_FileSystemTable_def
+#define USE_VFSTABLE_DEFINE
#include "ifilesystem.h"
-#include "iarchive.h"
-#include "eclasslib.h"
-#include "stream/stringstream.h"
-#include "stream/textfilestream.h"
-#include "modulesystem/moduleregistry.h"
-#include "os/path.h"
+
+#include "eclass_def.h"
+
+/*! \file eclass_def.cpp
+ \brief .def entity description format
+ implements parsing for .def entity format
+ this is statically linked into the radiant core as we always need it, but really considered
+ as an idependant module by the rest of the core. "ECLASS_MAJOR" "def"
+*/
+
+_QERScripLibTable g_ScripLibTable;
+_EClassManagerTable g_EClassManagerTable;
+_QERFuncTable_1 g_FuncTable;
+_QERFileSystemTable g_FileSystemTable_def;
+
+CSynapseBuiltinClientDef eclass_def;
+
+// forward declare, I'm cheap
+void Eclass_ScanFile (char *filename);
const char* EClass_GetExtension()
{
return "def";
}
-void Eclass_ScanFile (EntityClassCollector& collector, const char *filename);
-
-#include "modulesystem/singletonmodule.h"
-
-class EntityClassDefDependencies : public GlobalShaderCacheModuleRef, public GlobalScripLibModuleRef
+void CSynapseBuiltinClientDef::EnumerateInterfaces(CSynapseServer *server)
{
-};
+ AddAPI(SCRIPLIB_MAJOR, NULL, sizeof(g_ScripLibTable), SYN_REQUIRE, &g_ScripLibTable);
+ AddAPI(RADIANT_MAJOR, NULL, sizeof(g_FuncTable), SYN_REQUIRE, &g_FuncTable);
+ AddAPI(ECLASSMANAGER_MAJOR, NULL, sizeof(g_EClassManagerTable), SYN_REQUIRE, &g_EClassManagerTable);
+ // hardcode the minor for now, we can still add it to the synapse.config at some point
+ AddAPI(VFS_MAJOR, "pk3", sizeof(g_FileSystemTable_def), SYN_REQUIRE, &g_FileSystemTable_def);
+
+ AddAPI(ECLASS_MAJOR, "def", sizeof(_EClassTable));
+}
-class EclassDefAPI
+bool CSynapseBuiltinClientDef::RequestAPI(APIDescriptor_t *pAPI)
{
- EntityClassScanner m_eclassdef;
-public:
- typedef EntityClassScanner Type;
- STRING_CONSTANT(Name, "def");
-
- EclassDefAPI()
- {
- m_eclassdef.scanFile = &Eclass_ScanFile;
- m_eclassdef.getExtension = &EClass_GetExtension;
- }
- EntityClassScanner* getTable()
+ if (!strcmp(pAPI->major_name, ECLASS_MAJOR))
{
- return &m_eclassdef;
+ _EClassTable* pTable= static_cast<_EClassTable*>(pAPI->mpTable);
+ pTable->m_pfnScanFile = &Eclass_ScanFile;
+ pTable->m_pfnGetExtension = &EClass_GetExtension;
+
+ return true;
}
-};
-
-typedef SingletonModule<EclassDefAPI, EntityClassDefDependencies> EclassDefModule;
-typedef Static<EclassDefModule> StaticEclassDefModule;
-StaticRegisterModule staticRegisterEclassDef(StaticEclassDefModule::instance());
-
-
-#include "string/string.h"
-
-#include <stdlib.h>
-
-
-char com_token[1024];
-bool com_eof;
-
-/*
-==============
-COM_Parse
-
-Parse a token out of a string
-==============
-*/
-const char *COM_Parse (const char *data)
-{
- int c;
- int len;
-
- len = 0;
- com_token[0] = 0;
-
- if (!data)
- return 0;
-
-// skip whitespace
-skipwhite:
- while ( (c = *data) <= ' ')
- {
- if (c == 0)
- {
- com_eof = true;
- return 0; // end of file;
- }
- data++;
- }
-
-// skip // comments
- if (c=='/' && data[1] == '/')
- {
- while (*data && *data != '\n')
- data++;
- goto skipwhite;
- }
-
-
-// handle quoted strings specially
- if (c == '\"')
- {
- data++;
- do
- {
- c = *data++;
- if (c=='\"')
- {
- com_token[len] = 0;
- return data;
- }
- com_token[len] = c;
- len++;
- } while (1);
- }
-
-// parse single characters
- if (c=='{' || c=='}'|| c==')'|| c=='(' || c=='\'' || c==':')
- {
- com_token[len] = c;
- len++;
- com_token[len] = 0;
- return data+1;
- }
-
-// parse a regular word
- do
- {
- com_token[len] = c;
- data++;
- len++;
- c = *data;
- if (c=='{' || c=='}'|| c==')'|| c=='(' || c=='\'' || c==':')
- break;
- } while (c>32);
-
- com_token[len] = 0;
- return data;
+
+ Syn_Printf("ERROR: RequestAPI( '%s' ) not found in '%s'\n", pAPI->major_name, GetInfo());
+ return false;
}
-const char* Get_COM_Token()
+#include "version.h"
+
+const char* CSynapseBuiltinClientDef::GetInfo()
{
- return com_token;
+ return "Builtin .def module built " __DATE__ " " RADIANT_VERSION;
}
+// ------------------------------------------------------------------------------------------------
-const char *debugname;
+qboolean eclass_found;
+char *debugname;
-void setSpecialLoad(EntityClass *e, const char* pWhat, CopiedString& p)
+void setSpecialLoad(eclass_t *e, const char* pWhat, char*& p)
{
// Hydra: removed some amazingly bad cstring usage, whoever wrote that
// needs to be taken out and shot.
- const char *pText = 0;
- const char *where = 0;
+ char *pText = NULL;
+ char *where = NULL;
- where = strstr(e->comments(),pWhat);
+ p = NULL; // incase we don't find what we're looking for.
+ where = strstr(e->comments,pWhat);
if (!where)
return;
where = strchr(pText,'\"');
if (where)
{
- p = CopiedString(pText, where);
+ int len = (where-pText);
+ p = new char[len + 1];
+ strncpy(p,pText,len);
+ p[len]=0; // just to make sure, as most implementations of strncpy don't null terminate
}
else
- {
- p = pText;
- }
+ p = strdup(pText);
}
-#include "eclasslib.h"
-
-/*
-
-the classname, color triple, and bounding box are parsed out of comments
-A ? size means take the exact brush size.
-
-/ *QUAKED <classname> (0 0 0) ?
-/ *QUAKED <classname> (0 0 0) (-8 -8 -8) (8 8 8)
-
-Flag names can follow the size description:
-
-/ *QUAKED func_door (0 .5 .8) ? START_OPEN STONE_SOUND DOOR_DONT_LINK GOLD_KEY SILVER_KEY
-
-*/
-
-EntityClass *Eclass_InitFromText (const char *text)
+eclass_t *Eclass_InitFromText (char *text)
{
- EntityClass* e = Eclass_Alloc();
- e->free = &Eclass_Free;
+ char *t;
+ int len;
+ int r, i;
+ char parms[256], *p;
+ eclass_t *e;
+ char color[128];
+
+ e = (eclass_t*)malloc(sizeof(*e));
+ memset (e, 0, sizeof(*e));
+
+ text += strlen("/*QUAKED ");
// grab the name
text = COM_Parse (text);
- e->m_name = Get_COM_Token();
- debugname = e->name();
-
- {
- // grab the color, reformat as texture name
- int r = sscanf (text," (%f %f %f)", &e->color[0], &e->color[1], &e->color[2]);
- if (r != 3)
- return e;
- eclass_capture_state(e);
+ e->name = (char*)malloc (strlen(Get_COM_Token())+1);
+ strcpy (e->name, Get_COM_Token());
+ debugname = e->name;
+
+ // grab the color, reformat as texture name
+ r = sscanf (text," (%f %f %f)", &e->color[0], &e->color[1], &e->color[2]);
+ if (r != 3) {
+ return e;
}
+ sprintf (color, "(%f %f %f)", e->color[0], e->color[1], e->color[2]);
+ //strcpy (e->texdef.name, color);
+ e->texdef.SetName(color);
while (*text != ')')
{
if (!*text) {
- return 0;
+ return e;
}
text++;
}
if (Get_COM_Token()[0] == '(')
{ // parse the size as two vectors
e->fixedsize = true;
- int r = sscanf (text,"%f %f %f) (%f %f %f)", &e->mins[0], &e->mins[1], &e->mins[2],
+ r = sscanf (text,"%f %f %f) (%f %f %f)", &e->mins[0], &e->mins[1], &e->mins[2],
&e->maxs[0], &e->maxs[1], &e->maxs[2]);
if (r != 6) {
- return 0;
+ return e;
}
- for (int i=0 ; i<2 ; i++)
+ for (i=0 ; i<2 ; i++)
{
while (*text != ')')
{
if (!*text) {
- return 0;
+ return e;
}
text++;
}
}
}
- char parms[256];
// get the flags
- {
- // copy to the first /n
- char* p = parms;
- while (*text && *text != '\n')
- *p++ = *text++;
- *p = 0;
- text++;
- }
+ // copy to the first /n
+ p = parms;
+ while (*text && *text != '\n')
+ *p++ = *text++;
+ *p = 0;
+ text++;
+
+ // any remaining words are parm flags
+ p = parms;
+ for (i=0 ; i<MAX_FLAGS ; i++)
+ {
+ p = COM_Parse (p);
+ if (!p)
+ break;
+ strcpy (e->flagnames[i], Get_COM_Token());
+ }
+
+ // find the length until close comment
+ for (t=text ; t[0] && !(t[0]=='*' && t[1]=='/') ; t++)
+ ;
+
+ // copy the comment block out
+ len = t-text;
+ e->comments = (char*)malloc (len+1);
+ memcpy (e->comments, text, len);
+#ifdef _WIN32
+ // the win32 Gtk widgets are expecting text stuff to be in unix format (that is CR only instead of DOS's CR/LF)
+ // we convert on the fly by replacing the LF with a ' ' (yeah I'm cheap)
+ for (i=0 ; i<len ; i++)
+ if (text[i] == '\r')
+ e->comments[i] = ' ';
+ else
+ e->comments[i] = text[i];
+#endif
+ e->comments[len] = 0;
+
+ setSpecialLoad(e, "model=", e->modelpath);
+ setSpecialLoad(e, "skin=", e->skinpath);
+ char *pFrame = NULL;
+ setSpecialLoad(e, "frame=", pFrame);
+ if (pFrame != NULL)
{
- // any remaining words are parm flags
- const char* p = parms;
- for (std::size_t i=0 ; i<MAX_FLAGS ; i++)
- {
- p = COM_Parse (p);
- if (!p)
- break;
- strcpy (e->flagnames[i], Get_COM_Token());
- }
+ e->nFrame = atoi(pFrame);
+ delete pFrame; //Hydra - Fixed memory leak!
}
- e->m_comments = text;
+ if(!e->skinpath)
+ setSpecialLoad(e, "texture=", e->skinpath);
- setSpecialLoad(e, "model=", e->m_modelpath);
- StringOutputStream buffer(string_length(e->m_modelpath.c_str()));
- buffer << PathCleaned(e->m_modelpath.c_str());
- e->m_modelpath = buffer.c_str();
+ // setup show flags
+ e->nShowFlags = 0;
+ if (strcmpi(e->name, "light") == 0 || strcmpi(e->name, "dlight") == 0 || strcmpi(e->name, "lightjunior") == 0)
+ {
+ e->nShowFlags |= ECLASS_LIGHT;
+ }
- if(!e->fixedsize)
+ if ( (strnicmp(e->name, "info_player", strlen("info_player")) == 0)
+ ||(strnicmp(e->name, "path_corner", strlen("path_corner")) == 0)
+ ||(strnicmp(e->name, "team_ctf", strlen("team_ctf")) == 0)
+ ||(strnicmp(e->name, "misc_teleporter_dest", strlen("misc_teleporter_dest")) == 0)
+ )
{
- EntityClass_insertAttribute(*e, "angle", EntityClassAttribute("direction", "Direction", "0"));
+ e->nShowFlags |= ECLASS_ANGLE;
}
- else
+ if (strcmpi(e->name, "path") == 0)
+ {
+ e->nShowFlags |= ECLASS_PATH;
+ }
+ if (strcmpi(e->name, "misc_model") == 0)
{
- EntityClass_insertAttribute(*e, "angle", EntityClassAttribute("angle", "Yaw Angle", "0"));
+ e->nShowFlags |= ECLASS_MISCMODEL;
}
- EntityClass_insertAttribute(*e, "model", EntityClassAttribute("model", "Model"));
- EntityClass_insertAttribute(*e, "noise", EntityClassAttribute("sound", "Sound"));
+
return e;
}
-void Eclass_ScanFile (EntityClassCollector& collector, const char *filename)
+void Eclass_ScanFile (char *filename)
{
- EntityClass *e;
+ int size;
+ char *data;
+ eclass_t *e;
+ int i;
+ char temp[1024];
+
+ QE_ConvertDOSToUnixName( temp, filename );
- TextFileInputStream inputFile(filename);
- if(inputFile.failed())
+ size = vfsLoadFullPathFile(filename, (void**)&data);
+ if (size <= 0)
{
- globalErrorStream() << "ScanFile: " << filename << " not found\n";
+ Sys_FPrintf (SYS_ERR, "Eclass_ScanFile: %s not found\n", filename);
return;
}
- globalOutputStream() << "ScanFile: " << filename << "\n";
-
- enum EParserState
- {
- eParseDefault,
- eParseSolidus,
- eParseComment,
- eParseQuakeED,
- eParseEntityClass,
- eParseEntityClassEnd,
- } state = eParseDefault;
- const char* quakeEd = "QUAKED";
- const char* p = 0;
- StringBuffer buffer;
- SingleCharacterInputStream<TextFileInputStream> bufferedInput(inputFile);
- for(;;)
+ Sys_Printf ("ScanFile: %s\n", temp);
+ eclass_found = false;
+ for (i=0 ; i<size ; i++)
{
- char c;
- if(!bufferedInput.readChar(c))
+ if (!strncmp(data+i, "/*QUAKED",8))
{
- break;
- }
-
- switch(state)
- {
- case eParseDefault:
- if(c == '/')
- {
- state = eParseSolidus;
- }
- break;
- case eParseSolidus:
- if(c == '/')
- {
- state = eParseComment;
- }
- else if(c == '*')
- {
- p = quakeEd;
- state = eParseQuakeED;
- }
- break;
- case eParseComment:
- if(c == '\n')
- {
- state = eParseDefault;
- }
- break;
- case eParseQuakeED:
- if(c == *p)
- {
- if(*(++p) == '\0')
- {
- state = eParseEntityClass;
- }
- }
- else
- {
- state = eParseDefault;
- }
- break;
- case eParseEntityClass:
- if(c == '*')
- {
- state = eParseEntityClassEnd;
- }
- else
- {
- buffer.push_back(c);
- }
- break;
- case eParseEntityClassEnd:
- if(c == '/')
- {
- e = Eclass_InitFromText(buffer.c_str());
- state = eParseDefault;
- if (e)
- collector.insert(e);
- else
- globalErrorStream() << "Error parsing: " << debugname << " in " << filename << "\n";
-
- buffer.clear();
- state = eParseDefault;
- }
+ e = Eclass_InitFromText (data+i);
+ if (e)
+ Eclass_InsertAlphabetized (e);
else
- {
- buffer.push_back('*');
- buffer.push_back(c);
- state = eParseEntityClass;
- }
- break;
+ Sys_FPrintf (SYS_ERR, "Error parsing: %s in %s\n",debugname, filename);
+
+ // single ?
+ *Get_Eclass_E() = e;
+ Set_Eclass_Found(true);
+ if ( Get_Parsing_Single() )
+ break;
}
}
+
+ g_free (data);
}