gtkradiant/radiant/parse.cpp

223 lines
5.1 KiB
C++

/*
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 "stdafx.h"
char token[MAXTOKEN];
qboolean unget;
char* script_p;
int scriptline;
// Hydra: added support for GetTokenExtra()
const char *currentdelimiters;
qboolean script_keepdelimiter;
void StartTokenParsing( char *data ){
scriptline = 1;
script_p = data;
unget = false;
// Hydra: added support for GetTokenExtra()
currentdelimiters = NULL;
script_keepdelimiter = true;
}
qboolean GetToken( qboolean crossline ){
char *token_p;
if ( unget ) { // is a token already waiting?
unget = false;
return true;
}
//
// skip space
//
skipspace:
while ( *script_p <= 32 )
{
if ( !*script_p ) {
if ( !crossline ) {
Sys_FPrintf( SYS_WRN, "Warning: Line %i is incomplete [01]\n",scriptline );
}
return false;
}
if ( *script_p++ == '\n' ) {
if ( !crossline ) {
Sys_FPrintf( SYS_WRN, "Warning: Line %i is incomplete [02]\n",scriptline );
}
scriptline++;
}
}
if ( script_p[0] == '/' && script_p[1] == '/' ) { // comment field
if ( !crossline ) {
Sys_FPrintf( SYS_WRN, "Warning: Line %i is incomplete [03]\n",scriptline );
}
while ( *script_p++ != '\n' )
if ( !*script_p ) {
if ( !crossline ) {
Sys_FPrintf( SYS_WRN, "Warning: Line %i is incomplete [04]\n",scriptline );
}
return false;
}
scriptline++; // Hydra: fixed bad line numbers problem
goto skipspace;
}
//
// copy token
//
token_p = token;
if ( *script_p == '"' ) {
script_p++;
while ( *script_p != '"' )
{
if ( !*script_p ) {
Error( "EOF inside quoted token" );
}
*token_p++ = *script_p++;
if ( token_p == &token[MAXTOKEN] ) {
Error( "Token too large on line %i",scriptline );
}
}
script_p++;
}
else{
while ( *script_p > 32 )
{
// Hydra: added support for GetTokenExtra(), care was taken to maintain speed
if ( ( currentdelimiters ) && ( !script_keepdelimiter ) && ( strchr( currentdelimiters,*( script_p ) ) ) ) {
break;
}
*token_p++ = *script_p++;
if ( token_p == &token[MAXTOKEN] ) {
Error( "Token too large on line %i",scriptline );
}
// Hydra: added support for GetTokenExtra()
if ( ( currentdelimiters ) && ( strchr( currentdelimiters,*( script_p - 1 ) ) ) ) {
break;
}
}
}
*token_p = 0;
return true;
}
void UngetToken( void ){
unget = true;
}
/*
==============
GetTokenExtra
This function expands the use of GetToken() so it can be used to parse
more complex file formats.
Hydra - Notes:
You can use this function to split a string like this
string1:("string2")
into two strings, like this:
string1
string2
whilst still checking for the brackets and colons, like this:
GetTokenExtra(false,":",false);// contains "string1"
GetTokenExtra(false,":",true); // contains ":"
GetTokenExtra(false,"(",true); // contains "("
GetToken(false); // contains "string2"
GetTokenExtra(false,")",true); // contains ")"
here's what you get, given the same string, with this code:
GetToken(false); // contains "string1:("string2")"
Parsing will end if any character in the script matches any one of the
characters in the "delimiters" string.
it's also possible to do things like this:
source strings:
1,2
1:2
1-2
1*2
code:
GetTokenExtra(false,",:-*",false); // token contains "1"
GetTokenExtra(false,",:-*",false); // token contains the delimiter that was used
GetToken(false); // contains "2"
==============
*/
qboolean GetTokenExtra( qboolean crossline,const char *delimiters, qboolean keepdelimiter ){
qboolean result;
const char *olddelimiters = currentdelimiters; // store it
currentdelimiters = delimiters; // change the delimiters
script_keepdelimiter = keepdelimiter; // change the global flag
result = GetToken( crossline );
currentdelimiters = olddelimiters; // restore it
return( result );
}
/*
==============
TokenAvailable
Returns true if there is another token on the line
==============
*/
qboolean TokenAvailable( void ){
char *search_p;
search_p = script_p;
while ( *search_p <= 32 )
{
if ( *search_p == '\n' ) {
return false;
}
if ( *search_p == 0 ) {
return false;
}
search_p++;
}
if ( *search_p == ';' ) {
return false;
}
return true;
}