2011-04-27 11:58:34 +00:00
|
|
|
/*
|
2013-08-18 14:16:15 +00:00
|
|
|
* Copyright 2011-2013 Blender Foundation
|
2011-04-27 11:58:34 +00:00
|
|
|
*
|
2013-08-18 14:16:15 +00:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2011-04-27 11:58:34 +00:00
|
|
|
*
|
2013-08-18 14:16:15 +00:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2011-04-27 11:58:34 +00:00
|
|
|
*
|
2013-08-18 14:16:15 +00:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
2014-12-25 01:50:24 +00:00
|
|
|
* limitations under the License.
|
2011-04-27 11:58:34 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "util_foreach.h"
|
|
|
|
#include "util_string.h"
|
2016-02-05 08:09:39 +00:00
|
|
|
#include "util_windows.h"
|
2011-04-27 11:58:34 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
2016-02-05 08:09:39 +00:00
|
|
|
# ifndef vsnprintf
|
|
|
|
# define vsnprintf _vsnprintf
|
|
|
|
# endif
|
|
|
|
#endif /* _WIN32 */
|
2011-04-27 11:58:34 +00:00
|
|
|
|
|
|
|
CCL_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
string string_printf(const char *format, ...)
|
|
|
|
{
|
|
|
|
vector<char> str(128, 0);
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
va_list args;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
va_start(args, format);
|
|
|
|
result = vsnprintf(&str[0], str.size(), format, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
if(result == -1) {
|
|
|
|
/* not enough space or formatting error */
|
|
|
|
if(str.size() > 65536) {
|
|
|
|
assert(0);
|
|
|
|
return string("");
|
|
|
|
}
|
|
|
|
|
|
|
|
str.resize(str.size()*2, 0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if(result >= (int)str.size()) {
|
|
|
|
/* not enough space */
|
|
|
|
str.resize(result + 1, 0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return string(&str[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool string_iequals(const string& a, const string& b)
|
|
|
|
{
|
|
|
|
if(a.size() == b.size()) {
|
|
|
|
for(size_t i = 0; i < a.size(); i++)
|
|
|
|
if(toupper(a[i]) != toupper(b[i]))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-30 14:54:48 +00:00
|
|
|
void string_split(vector<string>& tokens,
|
|
|
|
const string& str,
|
|
|
|
const string& separators,
|
|
|
|
bool skip_empty_tokens)
|
2011-04-27 11:58:34 +00:00
|
|
|
{
|
2016-02-05 08:09:39 +00:00
|
|
|
size_t token_start = 0, token_length = 0;
|
|
|
|
for(size_t i = 0; i < str.size(); ++i) {
|
|
|
|
const char ch = str[i];
|
|
|
|
if(separators.find(ch) == string::npos) {
|
|
|
|
/* Current character is not a separator,
|
|
|
|
* append it to token by increasing token length.
|
|
|
|
*/
|
|
|
|
++token_length;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Current character is a separator,
|
2016-07-30 14:54:48 +00:00
|
|
|
* append current token to the list.
|
2016-02-05 08:09:39 +00:00
|
|
|
*/
|
2016-07-30 14:54:48 +00:00
|
|
|
if(!skip_empty_tokens || token_length > 0) {
|
2016-02-05 08:09:39 +00:00
|
|
|
string token = str.substr(token_start, token_length);
|
|
|
|
tokens.push_back(token);
|
|
|
|
}
|
|
|
|
token_start = i + 1;
|
|
|
|
token_length = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Append token from the tail of the string if exists. */
|
|
|
|
if(token_length) {
|
|
|
|
string token = str.substr(token_start, token_length);
|
|
|
|
tokens.push_back(token);
|
|
|
|
}
|
|
|
|
}
|
2011-04-27 11:58:34 +00:00
|
|
|
|
2016-02-05 08:09:39 +00:00
|
|
|
bool string_startswith(const string& s, const char *start)
|
|
|
|
{
|
|
|
|
size_t len = strlen(start);
|
2011-04-27 11:58:34 +00:00
|
|
|
|
2016-02-05 08:09:39 +00:00
|
|
|
if(len > s.size())
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
return strncmp(s.c_str(), start, len) == 0;
|
2011-04-27 11:58:34 +00:00
|
|
|
}
|
|
|
|
|
2013-11-28 00:38:23 +00:00
|
|
|
bool string_endswith(const string& s, const char *end)
|
|
|
|
{
|
|
|
|
size_t len = strlen(end);
|
|
|
|
|
|
|
|
if(len > s.size())
|
|
|
|
return 0;
|
|
|
|
else
|
2014-02-03 02:55:26 +00:00
|
|
|
return strncmp(s.c_str() + s.size() - len, end, len) == 0;
|
2013-11-28 00:38:23 +00:00
|
|
|
}
|
|
|
|
|
2013-12-07 01:29:53 +00:00
|
|
|
string string_strip(const string& s)
|
|
|
|
{
|
|
|
|
string result = s;
|
|
|
|
result.erase(0, result.find_first_not_of(' '));
|
|
|
|
result.erase(result.find_last_not_of(' ') + 1);
|
|
|
|
return result;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-05-21 21:43:18 +00:00
|
|
|
void string_replace(string& haystack, const string& needle, const string& other)
|
|
|
|
{
|
2016-02-05 08:09:39 +00:00
|
|
|
size_t i = 0, index;
|
|
|
|
while((index = haystack.find(needle, i)) != string::npos) {
|
|
|
|
haystack.replace(index, needle.size(), other);
|
|
|
|
i = index + other.size();
|
|
|
|
}
|
2015-05-21 21:43:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string string_remove_trademark(const string &s)
|
|
|
|
{
|
|
|
|
string result = s;
|
|
|
|
string_replace(result, "(TM)", "");
|
|
|
|
string_replace(result, "(R)", "");
|
|
|
|
|
|
|
|
return string_strip(result);
|
|
|
|
}
|
|
|
|
|
2016-01-07 06:47:58 +00:00
|
|
|
string string_from_bool(bool var)
|
2016-01-07 00:38:25 +00:00
|
|
|
{
|
|
|
|
if(var)
|
|
|
|
return "True";
|
|
|
|
else
|
|
|
|
return "False";
|
|
|
|
}
|
|
|
|
|
2016-02-05 08:09:39 +00:00
|
|
|
/* Wide char strings helpers for Windows. */
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
|
|
|
wstring string_to_wstring(const string& str)
|
|
|
|
{
|
2016-03-23 12:58:31 +00:00
|
|
|
const int length_wc = MultiByteToWideChar(CP_UTF8,
|
2016-02-05 08:09:39 +00:00
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.length(),
|
|
|
|
NULL,
|
|
|
|
0);
|
2016-02-14 16:05:29 +00:00
|
|
|
wstring str_wc(length_wc, 0);
|
2016-03-23 12:58:31 +00:00
|
|
|
MultiByteToWideChar(CP_UTF8,
|
2016-02-05 08:09:39 +00:00
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.length(),
|
|
|
|
&str_wc[0],
|
|
|
|
length_wc);
|
|
|
|
return str_wc;
|
|
|
|
}
|
|
|
|
|
|
|
|
string string_from_wstring(const wstring& str)
|
|
|
|
{
|
2016-03-23 12:58:31 +00:00
|
|
|
int length_mb = WideCharToMultiByte(CP_UTF8,
|
2016-02-05 08:09:39 +00:00
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.size(),
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL, NULL);
|
2016-02-14 16:05:29 +00:00
|
|
|
string str_mb(length_mb, 0);
|
2016-03-23 12:58:31 +00:00
|
|
|
WideCharToMultiByte(CP_UTF8,
|
2016-02-05 08:09:39 +00:00
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.size(),
|
|
|
|
&str_mb[0],
|
|
|
|
length_mb,
|
|
|
|
NULL, NULL);
|
|
|
|
return str_mb;
|
|
|
|
}
|
|
|
|
|
2016-03-23 12:58:31 +00:00
|
|
|
string string_to_ansi(const string& str)
|
|
|
|
{
|
|
|
|
const int length_wc = MultiByteToWideChar(CP_UTF8,
|
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.length(),
|
|
|
|
NULL,
|
|
|
|
0);
|
|
|
|
wstring str_wc(length_wc, 0);
|
|
|
|
MultiByteToWideChar(CP_UTF8,
|
|
|
|
0,
|
|
|
|
str.c_str(),
|
|
|
|
str.length(),
|
|
|
|
&str_wc[0],
|
|
|
|
length_wc);
|
|
|
|
|
|
|
|
int length_mb = WideCharToMultiByte(CP_ACP,
|
|
|
|
0,
|
|
|
|
str_wc.c_str(),
|
|
|
|
str_wc.size(),
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
string str_mb(length_mb, 0);
|
|
|
|
WideCharToMultiByte(CP_ACP,
|
|
|
|
0,
|
|
|
|
str_wc.c_str(),
|
|
|
|
str_wc.size(),
|
|
|
|
&str_mb[0],
|
|
|
|
length_mb,
|
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
return str_mb;
|
|
|
|
}
|
|
|
|
|
2016-02-05 08:09:39 +00:00
|
|
|
#endif /* _WIN32 */
|
|
|
|
|
2016-05-29 22:02:05 +00:00
|
|
|
string string_human_readable_size(size_t size)
|
|
|
|
{
|
|
|
|
static const char suffixes[] = "BKMGTPEZY";
|
|
|
|
|
|
|
|
const char* suffix = suffixes;
|
|
|
|
size_t r = 0;
|
|
|
|
|
|
|
|
while(size >= 1024) {
|
|
|
|
r = size % 1024;
|
|
|
|
size /= 1024;
|
|
|
|
suffix++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(*suffix != 'B')
|
|
|
|
return string_printf("%.2f%c", double(size*1024+r)/1024.0, *suffix);
|
|
|
|
else
|
|
|
|
return string_printf("%zu", size);
|
|
|
|
}
|
|
|
|
|
|
|
|
string string_human_readable_number(size_t num)
|
|
|
|
{
|
2016-06-27 08:49:25 +00:00
|
|
|
if(num == 0) {
|
|
|
|
return "0";
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add thousands separators. */
|
2016-05-29 22:02:05 +00:00
|
|
|
char buf[32];
|
|
|
|
|
|
|
|
char* p = buf+31;
|
|
|
|
*p = '\0';
|
|
|
|
|
|
|
|
int i = -1;
|
|
|
|
while(num) {
|
|
|
|
if(++i && i % 3 == 0)
|
|
|
|
*(--p) = ',';
|
|
|
|
|
|
|
|
*(--p) = '0' + (num % 10);
|
|
|
|
|
|
|
|
num /= 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2011-04-27 11:58:34 +00:00
|
|
|
CCL_NAMESPACE_END
|
|
|
|
|