2008-12-10 16:49:53 +00:00
|
|
|
/*********************************************/
|
2009-11-23 15:16:50 +00:00
|
|
|
/* string.cpp */
|
|
|
|
/* some useful functions to handle strings */
|
2008-12-10 16:49:53 +00:00
|
|
|
/*********************************************/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
|
|
#include "fctsys.h"
|
2008-12-07 17:55:29 +00:00
|
|
|
#include "macros.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
#include "kicad_string.h"
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
|
2008-12-10 16:49:53 +00:00
|
|
|
/* read a double-quote delimited text from source and put it in in dest,
|
2008-12-08 18:56:43 +00:00
|
|
|
* read NbMaxChar bytes max
|
2008-12-10 16:49:53 +00:00
|
|
|
* return the char count read from source
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
int ReadDelimitedText( char* dest, char* source, int NbMaxChar )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
int ii, jj, flag = 0;
|
|
|
|
|
|
|
|
for( ii = 0, jj = 0; ii < NbMaxChar - 1; jj++, source++ )
|
|
|
|
{
|
|
|
|
if( *source == 0 )
|
2008-12-07 17:55:29 +00:00
|
|
|
break; /* E.O.L. */
|
2009-11-23 15:16:50 +00:00
|
|
|
if( *source == '"' ) /* delimiter is " */
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
|
|
|
if( flag )
|
2009-11-23 15:16:50 +00:00
|
|
|
break; /* End of delimited text */
|
|
|
|
flag = 1; /* First delimiter found. */
|
2007-12-01 05:53:52 +00:00
|
|
|
}
|
|
|
|
else if( flag )
|
|
|
|
{
|
|
|
|
*dest = *source; dest++; ii++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
*dest = 0; /* Null terminated */
|
2007-12-01 05:53:52 +00:00
|
|
|
return jj;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-08 18:56:43 +00:00
|
|
|
/* Remove training spaces in text
|
|
|
|
* return a pointer on the first non space char in text
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
char* StrPurge( char* text )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
char* ptspace;
|
|
|
|
|
|
|
|
if( text == NULL )
|
|
|
|
return NULL;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
while( ( *text <= ' ' ) && *text )
|
2007-12-01 05:53:52 +00:00
|
|
|
text++;
|
|
|
|
|
|
|
|
ptspace = text + strlen( text ) - 1;
|
2009-11-23 15:16:50 +00:00
|
|
|
while( ( *ptspace <= ' ' ) && *ptspace && ( ptspace >= text ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
|
|
|
*ptspace = 0; ptspace--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return text;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
/* Read lines from File
|
|
|
|
* Skip void lines and comments (starting by #)
|
|
|
|
* return the first non void line.
|
2009-11-23 15:16:50 +00:00
|
|
|
* increments *LineNum for each line
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
char* GetLine( FILE* File, char* Line, int* LineNum, int SizeLine )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
do {
|
|
|
|
if( fgets( Line, SizeLine, File ) == NULL )
|
|
|
|
return NULL;
|
|
|
|
if( LineNum )
|
|
|
|
*LineNum += 1;
|
|
|
|
} while( Line[0] == '#' || Line[0] == '\n' || Line[0] == '\r'
|
|
|
|
|| Line[0] == 0 );
|
|
|
|
|
|
|
|
strtok( Line, "\n\r" );
|
|
|
|
return Line;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
/* return in aBuffer the date and time
|
|
|
|
* time is the local time.
|
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
char* DateAndTime( char* aBuffer )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
wxString datetime;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
datetime = DateAndTime();
|
|
|
|
strcpy( aBuffer, CONV_TO_UTF8( datetime ) );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
return aBuffer;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-12-01 05:53:52 +00:00
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
|
2008-12-08 18:56:43 +00:00
|
|
|
/* return the date and time in a wxString
|
2008-12-07 17:55:29 +00:00
|
|
|
* note: does the same thing than strftime()
|
|
|
|
* time is the local time.
|
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
wxString DateAndTime()
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
wxString Line;
|
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
wxDateTime datetime = wxDateTime::Now();
|
|
|
|
|
|
|
|
datetime.SetCountry( wxDateTime::Country_Default );
|
|
|
|
Line = datetime.Format( wxDefaultDateTimeFormat, wxDateTime::Local );
|
2007-12-01 05:53:52 +00:00
|
|
|
|
|
|
|
return Line;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2008-12-07 17:55:29 +00:00
|
|
|
* sort() function
|
|
|
|
* Same as strncmp() but numbers in strings
|
|
|
|
* are compared according to the value, not the ascii value of each digit
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
int StrLenNumCmp( const wxChar* str1, const wxChar* str2, int NbMax )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
int i;
|
|
|
|
int nb1 = 0, nb2 = 0;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
if( ( str1 == NULL ) || ( str2 == NULL ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for( i = 0; i < NbMax; i++ )
|
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
if( isdigit( *str1 ) && isdigit( *str2 ) ) /* digit found */
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
|
|
|
nb1 = 0; nb2 = 0;
|
|
|
|
while( isdigit( *str1 ) )
|
|
|
|
{
|
|
|
|
nb1 = nb1 * 10 + *str1 - '0'; str1++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while( isdigit( *str2 ) )
|
|
|
|
{
|
|
|
|
nb2 = nb2 * 10 + *str2 - '0'; str2++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( nb1 < nb2 )
|
|
|
|
return -1;
|
|
|
|
if( nb1 > nb2 )
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( *str1 < *str2 )
|
|
|
|
return -1;
|
|
|
|
if( *str1 > *str2 )
|
|
|
|
return 1;
|
2009-11-23 15:16:50 +00:00
|
|
|
if( ( *str1 == 0 ) && ( *str2 == 0 ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
return 0;
|
|
|
|
str1++; str2++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2008-12-07 17:55:29 +00:00
|
|
|
* sort() function
|
|
|
|
* Same as stricmp() but numbers in strings
|
|
|
|
* are compared according to the value, not the ascii value of each digit
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
int StrNumICmp( const wxChar* str1, const wxChar* str2 )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
return StrLenNumICmp( str1, str2, 32735 );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2008-12-07 17:55:29 +00:00
|
|
|
* sort() function
|
|
|
|
* Same as strnicmp() but numbers in strings
|
|
|
|
* are compared according to the value, not the ascii value of each digit
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
int StrLenNumICmp( const wxChar* str1, const wxChar* str2, int NbMax )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
int i;
|
|
|
|
int nb1 = 0, nb2 = 0;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
if( ( str1 == NULL ) || ( str2 == NULL ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
for( i = 0; i < NbMax; i++ )
|
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
if( isdigit( *str1 ) && isdigit( *str2 ) ) /* find number */
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
|
|
|
nb1 = 0; nb2 = 0;
|
|
|
|
while( isdigit( *str1 ) )
|
|
|
|
{
|
|
|
|
nb1 = nb1 * 10 + *str1 - '0'; str1++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while( isdigit( *str2 ) )
|
|
|
|
{
|
|
|
|
nb2 = nb2 * 10 + *str2 - '0'; str2++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( nb1 < nb2 )
|
|
|
|
return -1;
|
|
|
|
if( nb1 > nb2 )
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( toupper( *str1 ) < toupper( *str2 ) )
|
|
|
|
return -1;
|
|
|
|
if( toupper( *str1 ) > toupper( *str2 ) )
|
|
|
|
return 1;
|
|
|
|
if( (*str1 == 0 ) && ( *str2 == 0 ) )
|
|
|
|
return 0;
|
|
|
|
str1++; str2++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-08 18:56:43 +00:00
|
|
|
/* compare a string to a pattern
|
|
|
|
* ( usual chars * and ? allowed).
|
|
|
|
* if case_sensitive == true, comparison is case sensitive
|
|
|
|
* return true if match else false
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
bool WildCompareString( const wxString& pattern, const wxString& string_to_tst,
|
|
|
|
bool case_sensitive )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
const wxChar* cp = NULL, * mp = NULL;
|
|
|
|
const wxChar* wild, * string;
|
|
|
|
wxString _pattern, _string_to_tst;
|
|
|
|
|
|
|
|
if( case_sensitive )
|
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
wild = pattern.GetData();
|
|
|
|
string = string_to_tst.GetData();
|
2007-12-01 05:53:52 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
_pattern = pattern;
|
|
|
|
_pattern.MakeUpper();
|
|
|
|
_string_to_tst = string_to_tst;
|
|
|
|
_string_to_tst.MakeUpper();
|
|
|
|
wild = _pattern.GetData();
|
|
|
|
string = _string_to_tst.GetData();
|
2007-12-01 05:53:52 +00:00
|
|
|
}
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
while( ( *string ) && ( *wild != '*' ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
if( ( *wild != *string ) && ( *wild != '?' ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
return FALSE;
|
|
|
|
wild++; string++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while( *string )
|
|
|
|
{
|
|
|
|
if( *wild == '*' )
|
|
|
|
{
|
2008-12-07 17:55:29 +00:00
|
|
|
if( !*++wild )
|
2007-12-01 05:53:52 +00:00
|
|
|
return 1;
|
|
|
|
mp = wild;
|
|
|
|
cp = string + 1;
|
|
|
|
}
|
2009-11-23 15:16:50 +00:00
|
|
|
else if( ( *wild == *string ) || ( *wild == '?' ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
{
|
|
|
|
wild++;
|
|
|
|
string++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
wild = mp;
|
|
|
|
string = cp++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while( *wild == '*' )
|
|
|
|
{
|
|
|
|
wild++;
|
|
|
|
}
|
|
|
|
|
2008-12-07 17:55:29 +00:00
|
|
|
return !*wild;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Converts a string used to compensate for internalization of printf().
|
|
|
|
* Generated floats with a comma instead of point.
|
|
|
|
* Obsolete: use SetLocaleTo_C_standard instead
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
char* to_point( char* Text )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
char* line = Text;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-12-01 05:53:52 +00:00
|
|
|
if( Text == NULL )
|
|
|
|
return NULL;
|
|
|
|
for( ; *Text != 0; Text++ )
|
|
|
|
{
|
|
|
|
if( *Text == ',' )
|
|
|
|
*Text = '.';
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-12-01 05:53:52 +00:00
|
|
|
return line;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
/* Convert string to upper case.
|
|
|
|
* Returns pointer to the converted string.
|
2007-12-01 05:53:52 +00:00
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
char* strupper( char* Text )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2007-12-01 05:53:52 +00:00
|
|
|
char* code = Text;
|
|
|
|
|
|
|
|
if( Text )
|
|
|
|
{
|
|
|
|
while( *code )
|
|
|
|
{
|
2009-11-23 15:16:50 +00:00
|
|
|
if( ( *code >= 'a' ) && ( *code <= 'z' ) )
|
2007-12-01 05:53:52 +00:00
|
|
|
*code += 'A' - 'a';
|
|
|
|
code++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Text;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
2010-08-03 02:13:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
int RefDesStringCompare( const wxString& strFWord, const wxString& strSWord )
|
|
|
|
{
|
|
|
|
// The different sections of the first string
|
|
|
|
wxString strFWordBeg, strFWordMid, strFWordEnd;
|
|
|
|
|
|
|
|
// The different sections of the second string
|
|
|
|
wxString strSWordBeg, strSWordMid, strSWordEnd;
|
|
|
|
|
|
|
|
int isEqual = 0; // The numerical results of a string compare
|
|
|
|
int iReturn = 0; // The variable that is being returned
|
|
|
|
|
|
|
|
long lFirstDigit = 0; /* The converted middle section of the first
|
|
|
|
*string */
|
|
|
|
long lSecondDigit = 0; /* The converted middle section of the second
|
|
|
|
*string */
|
|
|
|
|
|
|
|
// Split the two strings into separate parts
|
|
|
|
SplitString( strFWord, &strFWordBeg, &strFWordMid, &strFWordEnd );
|
|
|
|
SplitString( strSWord, &strSWordBeg, &strSWordMid, &strSWordEnd );
|
|
|
|
|
|
|
|
// Compare the Beginning section of the strings
|
|
|
|
isEqual = strFWordBeg.CmpNoCase( strSWordBeg );
|
|
|
|
|
|
|
|
if( isEqual > 0 )
|
|
|
|
iReturn = 1;
|
|
|
|
else if( isEqual < 0 )
|
|
|
|
iReturn = -1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If the first sections are equal compare their digits
|
|
|
|
strFWordMid.ToLong( &lFirstDigit );
|
|
|
|
strSWordMid.ToLong( &lSecondDigit );
|
|
|
|
|
|
|
|
if( lFirstDigit > lSecondDigit )
|
|
|
|
iReturn = 1;
|
|
|
|
else if( lFirstDigit < lSecondDigit )
|
|
|
|
iReturn = -1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If the first two sections are equal compare the endings
|
|
|
|
isEqual = strFWordEnd.CmpNoCase( strSWordEnd );
|
|
|
|
|
|
|
|
if( isEqual > 0 )
|
|
|
|
iReturn = 1;
|
|
|
|
else if( isEqual < 0 )
|
|
|
|
iReturn = -1;
|
|
|
|
else
|
|
|
|
iReturn = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return iReturn;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int SplitString( wxString strToSplit,
|
|
|
|
wxString* strBeginning,
|
|
|
|
wxString* strDigits,
|
|
|
|
wxString* strEnd )
|
|
|
|
{
|
|
|
|
// Clear all the return strings
|
|
|
|
strBeginning->Empty();
|
|
|
|
strDigits->Empty();
|
|
|
|
strEnd->Empty();
|
|
|
|
|
|
|
|
// There no need to do anything if the string is empty
|
|
|
|
if( strToSplit.length() == 0 )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Starting at the end of the string look for the first digit
|
|
|
|
int ii;
|
|
|
|
for( ii = (strToSplit.length() - 1); ii >= 0; ii-- )
|
|
|
|
{
|
|
|
|
if( isdigit( strToSplit[ii] ) )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there were no digits then just set the single string
|
|
|
|
if( ii < 0 )
|
|
|
|
*strBeginning = strToSplit;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Since there is at least one digit this is the trailing string
|
|
|
|
*strEnd = strToSplit.substr( ii + 1 );
|
|
|
|
|
|
|
|
// Go to the end of the digits
|
|
|
|
int position = ii + 1;
|
|
|
|
for( ; ii >= 0; ii-- )
|
|
|
|
{
|
|
|
|
if( !isdigit( strToSplit[ii] ) )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If all that was left was digits, then just set the digits string
|
|
|
|
if( ii < 0 )
|
|
|
|
*strDigits = strToSplit.substr( 0, position );
|
|
|
|
|
|
|
|
/* We were only looking for the last set of digits everything else is
|
|
|
|
*part of the preamble */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*strDigits = strToSplit.substr( ii + 1, position - ii - 1 );
|
|
|
|
*strBeginning = strToSplit.substr( 0, ii + 1 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|