2002-09-08 01:08:12 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2005-01-01 16:09:25 +00:00
|
|
|
* Copyright (C) 2002-2005 The ScummVM project
|
2002-09-08 01:08:12 +00:00
|
|
|
*
|
|
|
|
* 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "stdafx.h"
|
2003-08-01 12:21:04 +00:00
|
|
|
#include "common/str.h"
|
2002-09-08 01:08:12 +00:00
|
|
|
|
2002-10-08 00:11:41 +00:00
|
|
|
#include <ctype.h>
|
2002-09-26 11:44:02 +00:00
|
|
|
|
2003-10-02 17:43:02 +00:00
|
|
|
namespace Common {
|
2002-09-08 01:08:12 +00:00
|
|
|
|
2003-10-06 23:19:01 +00:00
|
|
|
const String String::emptyString;
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
String::String(const char *str, int len)
|
|
|
|
: _str(0), _len(0) {
|
2002-09-08 01:08:12 +00:00
|
|
|
_refCount = new int(1);
|
2002-09-26 12:29:10 +00:00
|
|
|
if (str && len != 0) {
|
|
|
|
if (len > 0)
|
2002-09-26 11:44:02 +00:00
|
|
|
_capacity = _len = len;
|
|
|
|
else
|
2002-09-28 19:25:09 +00:00
|
|
|
_capacity = _len = strlen(str);
|
2002-09-08 01:08:12 +00:00
|
|
|
_str = (char *)calloc(1, _capacity+1);
|
2002-09-26 11:44:02 +00:00
|
|
|
memcpy(_str, str, _len);
|
|
|
|
_str[_len] = 0;
|
2002-09-08 01:08:12 +00:00
|
|
|
} else {
|
|
|
|
_capacity = _len = 0;
|
|
|
|
_str = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
String::String(const String &str)
|
|
|
|
: _str(0), _len(0) {
|
2002-09-08 01:08:12 +00:00
|
|
|
++(*str._refCount);
|
|
|
|
|
|
|
|
_refCount = str._refCount;
|
|
|
|
_capacity = str._capacity;
|
2002-11-21 16:51:33 +00:00
|
|
|
_len = str._len;
|
2002-09-08 01:08:12 +00:00
|
|
|
_str = str._str;
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String::~String() {
|
2002-09-08 01:08:12 +00:00
|
|
|
decRefCount();
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::decRefCount() {
|
2002-09-08 01:08:12 +00:00
|
|
|
--(*_refCount);
|
|
|
|
if (*_refCount <= 0) {
|
|
|
|
delete _refCount;
|
2004-03-25 11:25:50 +00:00
|
|
|
free(_str);
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String& String::operator =(const char *str) {
|
2002-09-28 19:25:09 +00:00
|
|
|
int len = strlen(str);
|
2002-09-08 01:08:12 +00:00
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(len, false);
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 01:08:12 +00:00
|
|
|
_len = len;
|
|
|
|
memcpy(_str, str, _len + 1);
|
|
|
|
} else if (_len > 0) {
|
|
|
|
decRefCount();
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 01:08:12 +00:00
|
|
|
_refCount = new int(1);
|
|
|
|
_capacity = 0;
|
|
|
|
_len = 0;
|
|
|
|
_str = 0;
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String &String::operator =(const String &str) {
|
2002-09-08 01:08:12 +00:00
|
|
|
++(*str._refCount);
|
|
|
|
|
|
|
|
decRefCount();
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 01:08:12 +00:00
|
|
|
_refCount = str._refCount;
|
|
|
|
_capacity = str._capacity;
|
|
|
|
_len = str._len;
|
|
|
|
_str = str._str;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2004-07-21 14:20:37 +00:00
|
|
|
String& String::operator =(char c) {
|
|
|
|
ensureCapacity(1, false);
|
|
|
|
_len = 1;
|
|
|
|
_str[0] = c;
|
|
|
|
_str[1] = 0;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String &String::operator +=(const char *str) {
|
2002-09-28 19:25:09 +00:00
|
|
|
int len = strlen(str);
|
2002-09-08 01:08:12 +00:00
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_len + len, true);
|
|
|
|
|
|
|
|
memcpy(_str + _len, str, len + 1);
|
|
|
|
_len += len;
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String &String::operator +=(const String &str) {
|
2002-09-08 01:08:12 +00:00
|
|
|
int len = str._len;
|
|
|
|
if (len > 0) {
|
|
|
|
ensureCapacity(_len + len, true);
|
|
|
|
|
|
|
|
memcpy(_str + _len, str._str, len + 1);
|
|
|
|
_len += len;
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
String &String::operator += (char c) {
|
2002-09-08 01:08:12 +00:00
|
|
|
ensureCapacity(_len + 1, true);
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 01:08:12 +00:00
|
|
|
_str[_len++] = c;
|
|
|
|
_str[_len] = 0;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2005-02-06 19:00:59 +00:00
|
|
|
bool String::hasPrefix(const char *x) const {
|
|
|
|
assert(x != 0);
|
|
|
|
// Compare x with the start of _str.
|
|
|
|
const char *y = c_str();
|
|
|
|
while (*x && *x == *y) {
|
|
|
|
++x;
|
|
|
|
++y;
|
|
|
|
}
|
|
|
|
// It's a prefix, if and only if all letters in x are 'used up' before
|
|
|
|
// _str ends.
|
|
|
|
return *x == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool String::hasSuffix(const char *x) const {
|
|
|
|
assert(x != 0);
|
|
|
|
// Compare x with the end of _str.
|
|
|
|
const int x_len = strlen(x);
|
|
|
|
if (x_len > _len)
|
|
|
|
return false;
|
|
|
|
const char *y = c_str() + _len - x_len;
|
|
|
|
while (*x && *x == *y) {
|
|
|
|
++x;
|
|
|
|
++y;
|
|
|
|
}
|
|
|
|
// It's a suffix, if and only if all letters in x are 'used up' before
|
|
|
|
// _str ends.
|
|
|
|
return *x == 0;
|
|
|
|
}
|
|
|
|
|
2002-09-08 11:46:42 +00:00
|
|
|
void String::deleteLastChar() {
|
|
|
|
if (_len > 0) {
|
|
|
|
ensureCapacity(_len - 1, true);
|
|
|
|
_str[--_len] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::deleteChar(int p) {
|
2003-01-10 21:33:42 +00:00
|
|
|
if (p >= 0 && p < _len) {
|
|
|
|
ensureCapacity(_len - 1, true);
|
|
|
|
while (p++ < _len)
|
|
|
|
_str[p-1] = _str[p];
|
|
|
|
_len--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::clear() {
|
2002-09-08 11:46:42 +00:00
|
|
|
if (_capacity) {
|
|
|
|
decRefCount();
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 11:46:42 +00:00
|
|
|
_refCount = new int(1);
|
|
|
|
_capacity = 0;
|
|
|
|
_len = 0;
|
|
|
|
_str = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::insertChar(char c, int p) {
|
2004-02-14 01:12:35 +00:00
|
|
|
// FIXME: This should be an 'assert', not an 'if' !
|
2003-01-10 21:33:42 +00:00
|
|
|
if (p >= 0 && p <= _len) {
|
2004-02-14 01:12:35 +00:00
|
|
|
ensureCapacity(_len + 1, true);
|
|
|
|
_len++;
|
2003-01-10 21:33:42 +00:00
|
|
|
for (int i = _len; i > p; i--) {
|
|
|
|
_str[i] = _str[i-1];
|
|
|
|
}
|
|
|
|
_str[p] = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::toLowercase() {
|
2002-10-08 00:11:41 +00:00
|
|
|
if (_str == 0 || _len == 0)
|
|
|
|
return;
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-10-08 00:11:41 +00:00
|
|
|
for (int i = 0; i < _len; ++i)
|
|
|
|
_str[i] = tolower(_str[i]);
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::toUppercase() {
|
2002-10-08 00:11:41 +00:00
|
|
|
if (_str == 0 || _len == 0)
|
|
|
|
return;
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-10-08 00:11:41 +00:00
|
|
|
for (int i = 0; i < _len; ++i)
|
|
|
|
_str[i] = toupper(_str[i]);
|
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
void String::ensureCapacity(int new_len, bool keep_old) {
|
2002-09-08 11:46:42 +00:00
|
|
|
// If there is not enough space, or if we are not the only owner
|
|
|
|
// of the current data, then we have to reallocate it.
|
|
|
|
if (new_len <= _capacity && *_refCount == 1)
|
|
|
|
return;
|
|
|
|
|
2003-11-08 22:43:46 +00:00
|
|
|
int newCapacity = (new_len <= _capacity) ? _capacity : new_len + 32;
|
|
|
|
char *newStr = (char *)calloc(1, newCapacity+1);
|
2002-09-08 11:46:42 +00:00
|
|
|
|
|
|
|
if (keep_old && _str)
|
|
|
|
memcpy(newStr, _str, _len + 1);
|
|
|
|
else
|
|
|
|
_len = 0;
|
|
|
|
|
|
|
|
decRefCount();
|
2003-03-06 16:27:06 +00:00
|
|
|
|
2002-09-08 11:46:42 +00:00
|
|
|
_refCount = new int(1);
|
|
|
|
_capacity = newCapacity;
|
|
|
|
_str = newStr;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator ==(const String &x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
return (0 == strcmp(c_str(), x.c_str()));
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator ==(const char *x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
assert(x != 0);
|
|
|
|
return (0 == strcmp(c_str(), x));
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator !=(const String &x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
return (0 != strcmp(c_str(), x.c_str()));
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator !=(const char *x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
assert(x != 0);
|
|
|
|
return (0 != strcmp(c_str(), x));
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator < (const String &x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
return strcmp(c_str(), x.c_str()) < 0;
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator <= (const String &x) const {
|
2003-11-07 00:02:03 +00:00
|
|
|
return strcmp(c_str(), x.c_str()) <= 0;
|
2002-09-08 01:08:12 +00:00
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator > (const String &x) const {
|
2002-09-08 01:08:12 +00:00
|
|
|
return (x < *this);
|
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool String::operator >= (const String &x) const {
|
2002-09-08 01:08:12 +00:00
|
|
|
return (x <= *this);
|
|
|
|
}
|
|
|
|
|
2004-06-27 23:58:41 +00:00
|
|
|
#pragma mark -
|
|
|
|
|
|
|
|
String operator +(const String &x, const String &y) {
|
|
|
|
String temp(x);
|
|
|
|
temp += y;
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
String operator +(const char *x, const String &y) {
|
|
|
|
String temp(x);
|
|
|
|
temp += y;
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
String operator +(const String &x, const char *y) {
|
|
|
|
String temp(x);
|
|
|
|
temp += y;
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark -
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool operator == (const char* y, const String &x) {
|
2002-09-08 01:08:12 +00:00
|
|
|
return (x == y);
|
|
|
|
}
|
|
|
|
|
2005-01-15 21:42:59 +00:00
|
|
|
bool operator != (const char* y, const String &x) {
|
2002-09-08 01:08:12 +00:00
|
|
|
return x != y;
|
|
|
|
}
|
|
|
|
|
2003-10-02 17:43:02 +00:00
|
|
|
} // End of namespace Common
|