Files
commandergenius/src/StringUtils.h

371 lines
12 KiB
C++

/*
OpenLieroX
string utilities
code under LGPL
created 01-05-2007
by Albert Zeyer and Dark Charlie
*/
#ifndef __STRINGUTILS_H__
#define __STRINGUTILS_H__
#include <SDL.h> // for Uint32
#include <cstdio> // for FILE
#include <string>
#include <sstream>
#include <vector>
#include <cassert>
#include <list>
#include <limits.h>
#include "Color.h" // for StrToCol
#include "Iterator.h"
//
// C-string handling routines
//
// HINT: these are obsolete, use std::string where possible!!!
// Secure c-string handling macros
// WARNING: don't use expressions like buf[i++] with the macros, because the "i" variable will be incremented twice in some macros!
#define fix_markend(chrarray) \
chrarray[sizeof(chrarray)-1] = '\0';
#define fix_strnlen(chrarray) \
strnlen(chrarray,sizeof(chrarray))
#define fix_strncpy(chrarray, src) \
{ strncpy(chrarray, src, sizeof(chrarray)); \
chrarray[sizeof(chrarray)-1] = '\0'; }
#define fix_strncat(chrarray, src) \
{ size_t destlen = strnlen(chrarray, sizeof(chrarray)); \
strncpy(&chrarray[destlen], src, sizeof(chrarray)-destlen); \
chrarray[sizeof(chrarray)-1] = '\0'; }
#define dyn_markend(dest, len) \
dest[len-1] = '\0';
#define dyn_strncpy(dest, src, len) \
{ strncpy(dest, src, len); \
dest[len-1] = '\0'; }
#define dyn_strncat(dest, src, len) \
{ size_t destlen = strnlen(dest, len); \
strncpy(&dest[destlen], src, len-destlen); \
dest[len-1] = '\0'; }
// Strnlen definition for compilers that don't have it
#if !defined(__USE_GNU) && _MSC_VER <= 1200
inline size_t strnlen(const char *str, size_t maxlen) {
register size_t i;
for(i = 0; (i < maxlen) && str[i]; ++i) {}
return i;
}
#endif
// Misc cross-compiler compatibility problem solutions
#ifdef WIN32
#if (defined(_MSC_VER) && (_MSC_VER <= 1200))
inline int strncasecmp(const char *str1, const char *str2, size_t l) {
return _strnicmp(str1, str2, l);
}
#endif
# define vsnprintf _vsnprintf
# define snprintf _snprintf
# define stricmp _stricmp
# define fcloseall _fcloseall
# define strcasecmp stricmp
#else
inline void strlwr(char* string) {
if(string)
while( *string ) {
*string = (char)tolower( *string );
string++;
}
}
#endif
/////////////
// Case-insensitive comparison of two chars, behaves like stringcasecmp
int chrcasecmp(const char c1, const char c2);
/////////////
// C-string itoa for non-windows compilers (on Windows it's defined in windows.h)
#ifndef WIN32
// TODOL remove this
inline char* itoa(int val, char* buf, int base) {
int i = 29; // TODO: bad style
buf[i+1] = '\0';
do {
buf = "0123456789abcdefghijklmnopqrstuvwxyz"[val % base] + buf;
--i, val /= base;
} while(val && i);
return &buf[i+1];
}
// Cross-compiler compatibility
# define stricmp strcasecmp
#endif
//
// C++ string (std::string) routines
//
// HINT: use these where possible
void TrimSpaces(std::string& szLine);
bool replace(const std::string& text, const std::string& what, const std::string& with, std::string& result);
bool replace(std::string& text, const std::string& what, const std::string& with);
std::string replacemax(const std::string& text, const std::string& what, const std::string& with, std::string& result, int max);
std::string replacemax(const std::string& text, const std::string& what, const std::string& with, int max);
std::string strip(const std::string& text, int width);
bool stripdot(std::string& text, int width);
void ucfirst(std::string& text);
std::string ReadUntil(const std::string& text, char until_character = '\n'); // will return whole text if not found
std::string ReadUntil(const std::string& text, std::string::const_iterator& start, char until_character, const std::string& alternative = "");
std::string ReadUntil(FILE* fp, char until_character = '\n');
Color StrToCol(const std::string& str);
Color StrToCol(const std::string& str, bool& fail);
std::vector<std::string> explode(const std::string& str, const std::string& delim);
void freadstr(std::string& result, size_t maxlen, FILE *fp);
size_t fwrite(const std::string& txt, size_t len, FILE* fp);
size_t findLastPathSep(const std::string& path);
void stringlwr(std::string& txt);
std::string stringtolower(const std::string& txt);
bool strincludes(const std::string& str, const std::string& what);
short stringcasecmp(const std::string& s1, const std::string& s2);
bool stringcaseequal(const std::string& s1, const std::string& s2);
bool subStrEqual(const std::string& s1, const std::string s2, size_t p);
bool subStrCaseEqual(const std::string& s1, const std::string s2, size_t p);
inline bool strStartsWith(const std::string& str, const std::string& start) { if(start.size() > str.size()) return false; return str.substr(0,start.size()) == start; }
inline bool strCaseStartsWith(const std::string& str, const std::string& start) { if(start.size() > str.size()) return false; return subStrCaseEqual(str,start,start.size()); }
size_t maxStartingEqualStr(const std::list<std::string>& strs);
size_t maxStartingCaseEqualStr(const std::list<std::string>& strs);
std::vector<std::string> splitstring(const std::string& str, size_t maxlen, size_t maxwidth, class CFont& font);
std::string splitStringWithNewLine(const std::string& str, size_t maxlen, size_t maxwidth, class CFont& font);
std::string GetFileExtension(const std::string& filename);
std::string GetBaseFilename(const std::string& filename);
std::string GetBaseFilenameWithoutExt(const std::string& filename);
std::list<std::string> SplitFilename(const std::string& filename, size_t numPartsFromRight = (size_t)-1); // splits fn by PathSep
std::string GetDirName(const std::string& filename);
size_t stringcasefind(const std::string& text, const std::string& search_for);
size_t stringcaserfind(const std::string& text, const std::string& search_for);
std::string GetNextWord(std::string::const_iterator it, const std::string& str);
std::string Base64Encode(const std::string &data);
std::string UrlEncode(const std::string &data); // Substitute space with + and all non-alphanum symbols with %XX
std::string AutoDetectLinks(const std::string& text);
std::string HtmlEntityUnpairedBrackets(const std::string &txt);
size_t GetPosByTextWidth(const std::string& text, int width, CFont *fnt);
std::string ColToHex(Color col);
std::string EscapeHtmlTags( const std::string & src ); // Escape all "<" and ">" and "&"
bool strSeemsLikeChatCommand(const std::string& str);
inline size_t subStrCount(const std::string& str, const std::string& substr) {
size_t c = 0, p = 0;
while((p = str.find(substr, p)) != std::string::npos) { c++; p++; }
return c;
}
typedef void (*PrintOutFct) (const std::string&);
inline void NullOut(const std::string&) {}
// returns true if last char was a newline
bool PrettyPrint(const std::string& prefix, const std::string& buf, PrintOutFct printOutFct, bool firstLineWithPrefix = true);
Iterator<char>::Ref HexDump(Iterator<char>::Ref start, PrintOutFct printOutFct, size_t mark = (size_t)-1, size_t count = (size_t)-1);
inline std::string FixedWidthStr_RightFill(const std::string& str, size_t w, char c) {
assert(str.size() <= w);
return str + std::string(str.size() - w, c);
}
inline std::string FixedWidthStr_LeftFill(const std::string& str, size_t w, char c) {
assert(str.size() <= w);
return std::string(w - str.size(), c) + str;
}
inline void StripQuotes(std::string& value) {
if( value.size() >= 2 )
if( value[0] == '"' && value[value.size()-1] == '"' )
value = value.substr( 1, value.size()-2 );
}
////////////////////
// Read a fixed-length C-string from a file
inline std::string freadfixedcstr(FILE *fp, size_t maxlen) {
std::string fileData;
freadstr(fileData, maxlen, fp);
return ReadUntil(fileData, '\0');
}
///////////////////
// Convert a numerical position to iterator
inline std::string::iterator PositionToIterator(std::string& str, size_t pos) {
std::string::iterator res = str.begin();
for (size_t i=0; i < pos && res != str.end(); ++i, res++) {}
return res;
}
// Conversion functions from string to numbers
template<typename T>
T from_string(const std::string& s, std::ios_base& (*f)(std::ios_base&), bool& failed) {
std::istringstream iss(s); T t = T();
failed = (iss >> f >> t).fail();
return t;
}
template<typename T>
T from_string(const std::string& s, std::ios_base& (*f)(std::ios_base&)) {
std::istringstream iss(s); T t = T();
iss >> f >> t;
return t;
}
template<typename T>
T from_string(const std::string& s, bool& failed) {
std::istringstream iss(s); T t = T();
failed = (iss >> t).fail();
return t;
}
// Conversion functions from numbers to string
template<typename T>
std::string to_string(T val) {
std::ostringstream oss;
oss << val;
return oss.str();
}
template<>
inline std::string to_string<bool>(bool val) {
if(val) return "true"; else return "false";
}
template<>
inline std::string to_string<const char*>(const char* val) {
if(val) return val; else return "";
}
template<>
inline bool from_string<bool>(const std::string& s, bool& fail) {
std::string s1(stringtolower(s));
TrimSpaces(s1);
if( s1 == "true" || s1 == "yes" || s1 == "on" ) return true;
else if( s1 == "false" || s1 == "no" || s1 == "off" ) return false;
return from_string<int>(s, fail) != 0;
}
template<> VectorD2<int> from_string< VectorD2<int> >(const std::string& s, bool& fail);
template<> inline std::string to_string< VectorD2<int> >(VectorD2<int> v) { return "(" + to_string(v.x) + "," + to_string(v.y) + ")"; }
template<typename T>
T from_string(const std::string& s) {
bool fail; return from_string<T>(s, fail);
}
inline int atoi(const std::string& str) { return from_string<int>(str); }
inline float atof(const std::string& str) { return from_string<float>(str); }
inline std::string ftoa(float val, int precision = -1)
{
std::string res = to_string<float>(val);
if (precision != -1) {
size_t dotpos = res.find_last_of('.');
if (dotpos == std::string::npos) {
res += '.';
for (int i = 0; i < precision; i++)
res += '0';
} else {
res = res.substr(0, dotpos + precision);
}
}
return res;
}
inline std::string itoa(unsigned long num, short base=10) {
std::string buf;
do {
buf = "0123456789abcdefghijklmnopqrstuvwxyz"[num % base] + buf;
num /= base;
} while(num);
return buf;
}
// std::string itoa
inline std::string itoa(long num, short base=10) {
if(num >= 0)
return itoa((unsigned long)num, base);
else
return "-" + itoa((unsigned long)-num, base);
}
inline std::string itoa(int num, short base=10) { return itoa((long)num,base); }
inline std::string itoa(unsigned int num, short base=10) { return itoa((unsigned long)num,base); }
// If 64-bit long available?
#ifdef ULLONG_MAX
inline std::string itoa(unsigned long long num, short base=10) {
std::string buf;
do {
buf = "0123456789abcdefghijklmnopqrstuvwxyz"[num % base] + buf;
num /= base;
} while(num);
return buf;
}
#endif
template<typename _T> std::string hex(_T num) { return itoa(num,16); }
struct simple_reversestring_hasher {
size_t operator() (const std::string& str) const {
std::string::const_reverse_iterator pos = str.rbegin();
unsigned short nibble = 0;
size_t result = 0;
for(; pos != str.rend() && nibble < sizeof(size_t)*2; pos++, nibble++)
result += ((size_t)*pos % 16) << nibble*4;
return result;
}
};
struct stringcaseless {
bool operator()(const std::string& s1, const std::string& s2) const {
return stringcasecmp(s1,s2) < 0;
}
};
struct const_string_iterator {
const std::string& str;
size_t pos;
const_string_iterator(const std::string& s, size_t p = 0) : str(s), pos(p) {}
const_string_iterator& operator++() { pos++; return *this; }
const_string_iterator& operator--() { assert(pos > 0); pos--; return *this; }
bool operator==(const const_string_iterator& i) const {
return &str == &i.str && (pos == i.pos || (pos > str.size() && i.pos > str.size()));
}
bool operator!=(const const_string_iterator& i) const { return !(*this == i); }
char operator*() const { return str[pos]; }
};
#endif