|
|
/*
* This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2004-2019 KiCad Developers, see AUTHORS.txt for contributors. * * This program 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. * * This program 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 this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
* or you may search the http://www.gnu.org website for the version 2 license,
* or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */
/**
* @file string.cpp * @brief Some useful functions to handle strings. */
#include <fctsys.h>
#include <macros.h>
#include <richio.h> // StrPrintf
#include <kicad_string.h>
/**
* Illegal file name characters used to insure file names will be valid on all supported * platforms. This is the list of illegal file name characters for Windows which includes * the illegal file name characters for Linux and OSX. */static const char illegalFileNameChars[] = "\\/:\"<>|";
wxString EscapeString( const wxString& aSource ){#if 1
return aSource;#else
wxString converted;
for( wxUniChar c: aSource ) { if( c == '\"' ) converted += """; else if( c == '\'' ) converted += "'"; else if( c == '&' ) converted += "&"; else if( c == '<' ) converted += "<"; else if( c == '>' ) converted += ">"; else if( c == '\\' ) converted += "∖"; else if( c == '/' ) converted += "⁄"; else if( c == '|' ) converted += "|"; else if( c == ':' ) converted += ":"; else if( c == ' ' ) converted += " "; else if( c == '%' ) converted += "%"; else if( c == '$' ) converted += "$"; else if( c == '\t' ) converted += "&tab;"; else if( c == '\n' || c == '\r' ) converted += "&Newline;"; else converted += c; }
return converted;#endif
}
wxString UnescapeString( const wxString& aSource ){#if 1
return aSource;#else
wxString converted = aSource;
converted.Replace( """, "\"" ); converted.Replace( "'", "'" ); converted.Replace( "<", "<" ); converted.Replace( ">", ">" ); converted.Replace( "∖", "\\" ); converted.Replace( "⁄", "/" ); converted.Replace( "|", "|" ); converted.Replace( ":", ":" ); converted.Replace( " ", " " ); converted.Replace( "%", "%" ); converted.Replace( "$", "$" ); converted.Replace( "&tab;", "\t" ); converted.Replace( "&Newline;", "\n" );
// must be done last
converted.Replace( "&", "&" );
return converted;#endif
}
int ReadDelimitedText( wxString* aDest, const char* aSource ){ std::string utf8; // utf8 but without escapes and quotes.
bool inside = false; const char* start = aSource; char cc;
while( (cc = *aSource++) != 0 ) { if( cc == '"' ) { if( inside ) break; // 2nd double quote is end of delimited text
inside = true; // first delimiter found, make note, do not copy
}
else if( inside ) { if( cc == '\\' ) { cc = *aSource++;
if( !cc ) break;
// do no copy the escape byte if it is followed by \ or "
if( cc != '"' && cc != '\\' ) utf8 += '\\';
utf8 += cc; } else { utf8 += cc; } } }
*aDest = FROM_UTF8( utf8.c_str() );
return aSource - start;}
int ReadDelimitedText( char* aDest, const char* aSource, int aDestSize ){ if( aDestSize <= 0 ) return 0;
bool inside = false; const char* start = aSource; char* limit = aDest + aDestSize - 1; char cc;
while( (cc = *aSource++) != 0 && aDest < limit ) { if( cc == '"' ) { if( inside ) break; // 2nd double quote is end of delimited text
inside = true; // first delimiter found, make note, do not copy
}
else if( inside ) { if( cc == '\\' ) { cc = *aSource++;
if( !cc ) break;
// do no copy the escape byte if it is followed by \ or "
if( cc != '"' && cc != '\\' ) *aDest++ = '\\';
if( aDest < limit ) *aDest++ = cc; } else { *aDest++ = cc; } } }
*aDest = 0;
return aSource - start;}
std::string EscapedUTF8( const wxString& aString ){ std::string utf8 = TO_UTF8( aString );
std::string ret;
ret += '"';
for( std::string::const_iterator it = utf8.begin(); it!=utf8.end(); ++it ) { // this escaping strategy is designed to be compatible with ReadDelimitedText():
if( *it == '"' ) { ret += '\\'; ret += '"'; } else if( *it == '\\' ) { ret += '\\'; // double it up
ret += '\\'; } else { ret += *it; } }
ret += '"';
return ret;}
wxString EscapedHTML( const wxString& aString ){ wxString converted;
for( wxUniChar c: aString ) { if( c == '\"' ) converted += """; else if( c == '\'' ) converted += "'"; else if( c == '&' ) converted += "&"; else if( c == '<' ) converted += "<"; else if( c == '>' ) converted += ">"; else converted += c; }
return converted;}
char* StrPurge( char* text ){ static const char whitespace[] = " \t\n\r\f\v";
if( text ) { while( *text && strchr( whitespace, *text ) ) ++text;
char* cp = text + strlen( text ) - 1;
while( cp >= text && strchr( whitespace, *cp ) ) *cp-- = '\0'; }
return text;}
char* GetLine( FILE* File, char* Line, int* LineNum, int SizeLine ){ 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;}
wxString DateAndTime(){ wxDateTime datetime = wxDateTime::Now();
datetime.SetCountry( wxDateTime::Country_Default ); return datetime.Format( wxDefaultDateTimeFormat, wxDateTime::Local );}
int StrNumCmp( const wxString& aString1, const wxString& aString2, bool aIgnoreCase ){ int nb1 = 0, nb2 = 0;
auto str1 = aString1.begin(); auto str2 = aString2.begin();
while( str1 != aString1.end() && str2 != aString2.end() ) { wxUniChar c1 = *str1; wxUniChar c2 = *str2;
if( wxIsdigit( c1 ) && wxIsdigit( c2 ) ) // Both characters are digits, do numeric compare.
{ nb1 = 0; nb2 = 0;
do { c1 = *str1; nb1 = nb1 * 10 + (int) c1 - '0'; ++str1; } while( str1 != aString1.end() && wxIsdigit( *str1 ) );
do { c2 = *str2; nb2 = nb2 * 10 + (int) c2 - '0'; ++str2; } while( str2 != aString2.end() && wxIsdigit( *str2 ) );
if( nb1 < nb2 ) return -1;
if( nb1 > nb2 ) return 1;
c1 = ( str1 != aString1.end() ) ? *str1 : wxUniChar( 0 ); c2 = ( str2 != aString2.end() ) ? *str2 : wxUniChar( 0 ); }
// Any numerical comparisons to here are identical.
if( aIgnoreCase ) { if( wxToupper( c1 ) < wxToupper( c2 ) ) return -1;
if( wxToupper( c1 ) > wxToupper( c2 ) ) return 1; } else { if( c1 < c2 ) return -1;
if( c1 > c2 ) return 1; }
if( str1 != aString1.end() ) ++str1;
if( str2 != aString2.end() ) ++str2; }
if( str1 == aString1.end() && str2 != aString2.end() ) { return -1; // Identical to here but aString1 is longer.
} else if( str1 != aString1.end() && str2 == aString2.end() ) { return 1; // Identical to here but aString2 is longer.
}
return 0;}
bool WildCompareString( const wxString& pattern, const wxString& string_to_tst, bool case_sensitive ){ const wxChar* cp = NULL, * mp = NULL; const wxChar* wild, * string; wxString _pattern, _string_to_tst;
if( case_sensitive ) { wild = pattern.GetData(); string = string_to_tst.GetData(); } else { _pattern = pattern; _pattern.MakeUpper(); _string_to_tst = string_to_tst; _string_to_tst.MakeUpper(); wild = _pattern.GetData(); string = _string_to_tst.GetData(); }
while( ( *string ) && ( *wild != '*' ) ) { if( ( *wild != *string ) && ( *wild != '?' ) ) return false;
wild++; string++; }
while( *string ) { if( *wild == '*' ) { if( !*++wild ) return 1; mp = wild; cp = string + 1; } else if( ( *wild == *string ) || ( *wild == '?' ) ) { wild++; string++; } else { wild = mp; string = cp++; } }
while( *wild == '*' ) { wild++; }
return !*wild;}
bool ApplyModifier( double& value, const wxString& aString ){ static const wxString modifiers( wxT( "pnumkKM" ) );
if( !aString.length() ) return false;
wxChar modifier; wxString units;
if( modifiers.Find( aString[ 0 ] ) >= 0 ) { modifier = aString[ 0 ]; units = aString.Mid( 1 ).Trim(); } else { modifier = ' '; units = aString.Mid( 0 ).Trim(); }
if( units.length() && !units.CmpNoCase( wxT( "F" ) ) && !units.CmpNoCase( wxT( "hz" ) ) && !units.CmpNoCase( wxT( "W" ) ) && !units.CmpNoCase( wxT( "V" ) ) && !units.CmpNoCase( wxT( "H" ) ) ) return false;
if( modifier == 'p' ) value *= 1.0e-12; if( modifier == 'n' ) value *= 1.0e-9; else if( modifier == 'u' ) value *= 1.0e-6; else if( modifier == 'm' ) value *= 1.0e-3; else if( modifier == 'k' || modifier == 'K' ) value *= 1.0e3; else if( modifier == 'M' ) value *= 1.0e6; else if( modifier == 'G' ) value *= 1.0e9;
return true;}
int ValueStringCompare( wxString strFWord, wxString strSWord ){ // Compare unescaped text
strFWord = UnescapeString( strFWord ); strSWord = UnescapeString( strSWord );
// The different sections of the two strings
wxString strFWordBeg, strFWordMid, strFWordEnd; wxString strSWordBeg, strSWordMid, strSWordEnd;
// Split the two strings into separate parts
SplitString( strFWord, &strFWordBeg, &strFWordMid, &strFWordEnd ); SplitString( strSWord, &strSWordBeg, &strSWordMid, &strSWordEnd );
// Compare the Beginning section of the strings
int isEqual = strFWordBeg.CmpNoCase( strSWordBeg );
if( isEqual > 0 ) return 1; else if( isEqual < 0 ) return -1; else { // If the first sections are equal compare their digits
double lFirstNumber = 0; double lSecondNumber = 0; bool endingIsModifier = false;
strFWordMid.ToDouble( &lFirstNumber ); strSWordMid.ToDouble( &lSecondNumber );
endingIsModifier |= ApplyModifier( lFirstNumber, strFWordEnd ); endingIsModifier |= ApplyModifier( lSecondNumber, strSWordEnd );
if( lFirstNumber > lSecondNumber ) return 1; else if( lFirstNumber < lSecondNumber ) return -1; // If the first two sections are equal and the endings are modifiers then compare them
else if( !endingIsModifier ) return strFWordEnd.CmpNoCase( strSWordEnd ); // Ran out of things to compare; they must match
else return 0; }}
int SplitString( wxString strToSplit, wxString* strBeginning, wxString* strDigits, wxString* strEnd ){ static const wxString separators( wxT( ".," ) );
// 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( wxIsdigit( 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( !wxIsdigit( strToSplit[ii] ) && separators.Find( strToSplit[ii] ) < 0 ) 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;}
int GetTrailingInt( const wxString& aStr ){ int number = 0; int base = 1;
// Trim and extract the trailing numeric part
int index = aStr.Len() - 1;
while( index >= 0 ) { const char chr = aStr.GetChar( index );
if( chr < '0' || chr > '9' ) break;
number += ( chr - '0' ) * base; base *= 10; index--; }
return number;}
wxString GetIllegalFileNameWxChars(){ return FROM_UTF8( illegalFileNameChars );}
bool ReplaceIllegalFileNameChars( std::string* aName, int aReplaceChar ){ bool changed = false; std::string result; result.reserve( aName->length() );
for( std::string::iterator it = aName->begin(); it != aName->end(); ++it ) { if( strchr( illegalFileNameChars, *it ) ) { if( aReplaceChar ) StrPrintf( &result, "%c", aReplaceChar ); else StrPrintf( &result, "%%%02x", *it );
changed = true; } else { result += *it; } }
if( changed ) *aName = result;
return changed;}
bool ReplaceIllegalFileNameChars( wxString& aName, int aReplaceChar ){ bool changed = false; wxString result; result.reserve( aName.Length() ); wxString illWChars = GetIllegalFileNameWxChars();
for( wxString::iterator it = aName.begin(); it != aName.end(); ++it ) { if( illWChars.Find( *it ) != wxNOT_FOUND ) { if( aReplaceChar ) result += aReplaceChar; else result += wxString::Format( "%%%02x", *it );
changed = true; } else { result += *it; } }
if( changed ) aName = result;
return changed;}
|