| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 |
- /*
- -----------------------------------------------------------------------------
- This source file is part of OGRE
- (Object-oriented Graphics Rendering Engine)
- For the latest info, see http://www.ogre3d.org/
- Copyright (c) 2000-2011 Torus Knot Software Ltd
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be included in
- all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- THE SOFTWARE.
- -----------------------------------------------------------------------------
- */
- #include "OgreString.h"
- #include "OgreStringVector.h"
- namespace Ogre {
- //-----------------------------------------------------------------------
- const String StringUtil::BLANK;
- //-----------------------------------------------------------------------
- void StringUtil::trim(String& str, bool left, bool right)
- {
- /*
- size_t lspaces, rspaces, len = length(), i;
- lspaces = rspaces = 0;
- if( left )
- {
- // Find spaces / tabs on the left
- for( i = 0;
- i < len && ( at(i) == ' ' || at(i) == '\t' || at(i) == '\r');
- ++lspaces, ++i );
- }
-
- if( right && lspaces < len )
- {
- // Find spaces / tabs on the right
- for( i = len - 1;
- i >= 0 && ( at(i) == ' ' || at(i) == '\t' || at(i) == '\r');
- rspaces++, i-- );
- }
- *this = substr(lspaces, len-lspaces-rspaces);
- */
- static const String delims = " \t\r";
- if(right)
- str.erase(str.find_last_not_of(delims)+1); // trim right
- if(left)
- str.erase(0, str.find_first_not_of(delims)); // trim left
- }
- //-----------------------------------------------------------------------
- StringVector StringUtil::split( const String& str, const String& delims, unsigned int maxSplits)
- {
- StringVector ret;
- // Pre-allocate some space for performance
- ret.reserve(maxSplits ? maxSplits+1 : 10); // 10 is guessed capacity for most case
- unsigned int numSplits = 0;
- // Use STL methods
- size_t start, pos;
- start = 0;
- do
- {
- pos = str.find_first_of(delims, start);
- if (pos == start)
- {
- // Do nothing
- start = pos + 1;
- }
- else if (pos == String::npos || (maxSplits && numSplits == maxSplits))
- {
- // Copy the rest of the string
- ret.push_back( str.substr(start) );
- break;
- }
- else
- {
- // Copy up to delimiter
- ret.push_back( str.substr(start, pos - start) );
- start = pos + 1;
- }
- // parse up to next real data
- start = str.find_first_not_of(delims, start);
- ++numSplits;
- } while (pos != String::npos);
- return ret;
- }
- //-----------------------------------------------------------------------
- StringVector StringUtil::tokenise( const String& str, const String& singleDelims, const String& doubleDelims, unsigned int maxSplits)
- {
- StringVector ret;
- // Pre-allocate some space for performance
- ret.reserve(maxSplits ? maxSplits+1 : 10); // 10 is guessed capacity for most case
- unsigned int numSplits = 0;
- String delims = singleDelims + doubleDelims;
- // Use STL methods
- size_t start, pos;
- char curDoubleDelim = 0;
- start = 0;
- do
- {
- if (curDoubleDelim != 0)
- {
- pos = str.find(curDoubleDelim, start);
- }
- else
- {
- pos = str.find_first_of(delims, start);
- }
- if (pos == start)
- {
- char curDelim = str.at(pos);
- if (doubleDelims.find_first_of(curDelim) != String::npos)
- {
- curDoubleDelim = curDelim;
- }
- // Do nothing
- start = pos + 1;
- }
- else if (pos == String::npos || (maxSplits && numSplits == maxSplits))
- {
- if (curDoubleDelim != 0)
- {
- //Missing closer. Warn or throw exception?
- }
- // Copy the rest of the string
- ret.push_back( str.substr(start) );
- break;
- }
- else
- {
- if (curDoubleDelim != 0)
- {
- curDoubleDelim = 0;
- }
- // Copy up to delimiter
- ret.push_back( str.substr(start, pos - start) );
- start = pos + 1;
- }
- if (curDoubleDelim == 0)
- {
- // parse up to next real data
- start = str.find_first_not_of(singleDelims, start);
- }
-
- ++numSplits;
- } while (pos != String::npos);
- return ret;
- }
- //-----------------------------------------------------------------------
- void StringUtil::toLowerCase(String& str)
- {
- std::transform(
- str.begin(),
- str.end(),
- str.begin(),
- tolower);
- }
- //-----------------------------------------------------------------------
- void StringUtil::toUpperCase(String& str)
- {
- std::transform(
- str.begin(),
- str.end(),
- str.begin(),
- toupper);
- }
- //-----------------------------------------------------------------------
- bool StringUtil::startsWith(const String& str, const String& pattern, bool lowerCase)
- {
- size_t thisLen = str.length();
- size_t patternLen = pattern.length();
- if (thisLen < patternLen || patternLen == 0)
- return false;
- String startOfThis = str.substr(0, patternLen);
- if (lowerCase)
- StringUtil::toLowerCase(startOfThis);
- return (startOfThis == pattern);
- }
- //-----------------------------------------------------------------------
- bool StringUtil::endsWith(const String& str, const String& pattern, bool lowerCase)
- {
- size_t thisLen = str.length();
- size_t patternLen = pattern.length();
- if (thisLen < patternLen || patternLen == 0)
- return false;
- String endOfThis = str.substr(thisLen - patternLen, patternLen);
- if (lowerCase)
- StringUtil::toLowerCase(endOfThis);
- return (endOfThis == pattern);
- }
- //-----------------------------------------------------------------------
- String StringUtil::standardisePath(const String& init)
- {
- String path = init;
- std::replace( path.begin(), path.end(), '\\', '/' );
- if( path[path.length() - 1] != '/' )
- path += '/';
- return path;
- }
- //-----------------------------------------------------------------------
- void StringUtil::splitFilename(const String& qualifiedName,
- String& outBasename, String& outPath)
- {
- String path = qualifiedName;
- // Replace \ with / first
- std::replace( path.begin(), path.end(), '\\', '/' );
- // split based on final /
- size_t i = path.find_last_of('/');
- if (i == String::npos)
- {
- outPath.clear();
- outBasename = qualifiedName;
- }
- else
- {
- outBasename = path.substr(i+1, path.size() - i - 1);
- outPath = path.substr(0, i+1);
- }
- }
- //-----------------------------------------------------------------------
- void StringUtil::splitBaseFilename(const Ogre::String& fullName,
- Ogre::String& outBasename, Ogre::String& outExtention)
- {
- size_t i = fullName.find_last_of(".");
- if (i == Ogre::String::npos)
- {
- outExtention.clear();
- outBasename = fullName;
- }
- else
- {
- outExtention = fullName.substr(i+1);
- outBasename = fullName.substr(0, i);
- }
- }
- // ----------------------------------------------------------------------------------------------------------------------------------------------
- void StringUtil::splitFullFilename( const Ogre::String& qualifiedName,
- Ogre::String& outBasename, Ogre::String& outExtention, Ogre::String& outPath )
- {
- Ogre::String fullName;
- splitFilename( qualifiedName, fullName, outPath );
- splitBaseFilename( fullName, outBasename, outExtention );
- }
- //-----------------------------------------------------------------------
- bool StringUtil::match(const String& str, const String& pattern, bool caseSensitive)
- {
- String tmpStr = str;
- String tmpPattern = pattern;
- if (!caseSensitive)
- {
- StringUtil::toLowerCase(tmpStr);
- StringUtil::toLowerCase(tmpPattern);
- }
- String::const_iterator strIt = tmpStr.begin();
- String::const_iterator patIt = tmpPattern.begin();
- String::const_iterator lastWildCardIt = tmpPattern.end();
- while (strIt != tmpStr.end() && patIt != tmpPattern.end())
- {
- if (*patIt == '*')
- {
- lastWildCardIt = patIt;
- // Skip over looking for next character
- ++patIt;
- if (patIt == tmpPattern.end())
- {
- // Skip right to the end since * matches the entire rest of the string
- strIt = tmpStr.end();
- }
- else
- {
- // scan until we find next pattern character
- while(strIt != tmpStr.end() && *strIt != *patIt)
- ++strIt;
- }
- }
- else
- {
- if (*patIt != *strIt)
- {
- if (lastWildCardIt != tmpPattern.end())
- {
- // The last wildcard can match this incorrect sequence
- // rewind pattern to wildcard and keep searching
- patIt = lastWildCardIt;
- lastWildCardIt = tmpPattern.end();
- }
- else
- {
- // no wildwards left
- return false;
- }
- }
- else
- {
- ++patIt;
- ++strIt;
- }
- }
- }
- // If we reached the end of both the pattern and the string, we succeeded
- if (patIt == tmpPattern.end() && strIt == tmpStr.end())
- {
- return true;
- }
- else
- {
- return false;
- }
- }
- //-----------------------------------------------------------------------
- const String StringUtil::replaceAll(const String& source, const String& replaceWhat, const String& replaceWithWhat)
- {
- String result = source;
- String::size_type pos = 0;
- while(1)
- {
- pos = result.find(replaceWhat,pos);
- if (pos == String::npos) break;
- result.replace(pos,replaceWhat.size(),replaceWithWhat);
- pos += replaceWithWhat.size();
- }
- return result;
- }
- }
|