5 #include "TObjString.h"
30 #ifdef __WITHOUT_TSTRING_TOKENIZE
39 TObjArray* KVString::Tokenize(
const TString& delim)
const
47 std::list < Int_t > splitIndex;
49 Int_t i, start, nrDiff = 0;
50 for (i = 0; i < delim.Length(); i++) {
52 while (start < Length()) {
53 Int_t pos = Index(delim(i), start);
56 splitIndex.push_back(pos);
62 splitIndex.push_back(Length());
67 TObjArray* arr =
new TObjArray();
71 std::list < Int_t >::const_iterator it;
73 for (it = splitIndex.begin(); it != splitIndex.end(); it++) {
75 for (it = splitIndex.begin();
76 it != (std::list < Int_t >::const_iterator) splitIndex.end();
80 if (stop - 1 >= start + 1) {
81 TString tok = (*this)(start + 1, stop - start - 1);
82 TObjString* objstr =
new TObjString(tok);
93 #ifdef __WITH_KVSTRING_ISDIGIT
101 Bool_t KVString::IsDigit()
const
108 const char* cp = Data();
109 Ssiz_t len = Length();
110 if (len == 0)
return kFALSE;
112 for (Ssiz_t i = 0; i < len; ++i) {
113 if (cp[i] !=
' ' && !isdigit(cp[i]))
return kFALSE;
114 if (cp[i] ==
' ') b++;
115 if (isdigit(cp[i])) d++;
124 #ifdef __WITH_KVSTRING_REMOVE
130 KVString& KVString::Remove(TString::EStripType st,
char c)
134 *
this = Strip(st, c);
140 #ifdef __WITH_KVSTRING_ATOI
147 Int_t KVString::Atoi()
const
153 Int_t end = Index(
" ");
161 tmp += (*this)(start, end - start);
163 end = Index(
" ", start);
166 tmp += (*this)(start, end - start);
167 return atoi(tmp.Data());
172 #ifdef __WITH_KVSTRING_ATOF
180 Double_t KVString::Atof()
const
187 Int_t comma = Index(
",");
188 Int_t end = Index(
" ");
190 if (comma == -1 && end == -1)
195 tmp.Replace(comma, 1,
".");
199 return atof(tmp.Data());
204 tmp2 += tmp(start, end - start);
206 end = tmp.Index(
" ", start);
209 tmp2 += tmp(start, end - start);
210 return atof(tmp2.Data());
215 #ifdef __WITH_KVSTRING_ISFLOAT
228 Bool_t KVString::IsFloat()
const
240 if (IsDigit())
return kTRUE;
245 Int_t i_dot, i_e, i_plus, i_minus, i_comma;
246 i_dot = i_e = i_plus = i_minus = i_comma = -1;
248 i_dot = tmp.First(
'.');
249 if (i_dot > -1) tmp.Replace(i_dot, 1,
" ", 1);
250 i_comma = tmp.First(
',');
251 if (i_comma > -1) tmp.Replace(i_comma, 1,
" ", 1);
252 i_e = tmp.First(
'e');
254 tmp.Replace(i_e, 1,
" ", 1);
257 i_e = tmp.First(
'E');
258 if (i_e > -1) tmp.Replace(i_e, 1,
" ", 1);
260 i_plus = tmp.First(
'+');
261 if (i_plus > -1) tmp.Replace(i_plus, 1,
" ", 1);
262 i_minus = tmp.First(
'-');
263 if (i_minus > -1) tmp.Replace(i_minus, 1,
" ", 1);
266 return tmp.IsDigit();
278 ReplaceAll(&c1, 1, &c2, 1);
283 #ifdef __WITH_KVSTRING_ISWHITESPACE
288 Bool_t KVString::IsWhitespace()
const
291 return (Length() == CountChar(
' '));
348 Int_t read_items = 0;
350 const char* cp = Data();
351 Int_t int_format_length_descriptor = 0;
352 Bool_t zero_padding = kFALSE;
354 while (fmt[fmt_index] !=
'\0') {
356 if (fmt[fmt_index] ==
'%') {
359 if (fmt[fmt_index] >=
'0' && fmt[fmt_index] <=
'9') {
361 zero_padding = (fmt[fmt_index] ==
'0');
362 if (zero_padding) fmt_index++;
365 while (fmt[fmt_index] >=
'0' && fmt[fmt_index] <=
'9') {
366 length_of_number += fmt[fmt_index++];
368 int_format_length_descriptor = length_of_number.Atoi();
370 if (fmt[fmt_index] ==
'd') {
373 if (int_format_length_descriptor) {
378 Int_t figures_read = 0;
379 while (cp[str_index] >=
'0' && cp[str_index] <=
'9') {
380 dummy += cp[str_index++];
383 if (figures_read != int_format_length_descriptor) {
390 *(va_arg(args,
int*)) = dummy.Atoi();
402 Bool_t no_more_whitespace = kFALSE;
403 while (int_format_length_descriptor) {
404 if (cp[str_index] ==
'\0') {
409 if ((cp[str_index] !=
' ') && (cp[str_index] <
'0' || cp[str_index] >
'9')) {
414 if ((cp[str_index] ==
' ') && no_more_whitespace) {
419 if (cp[str_index] !=
' ') {
420 no_more_whitespace = kTRUE;
421 dummy += cp[str_index];
424 int_format_length_descriptor--;
427 if (!no_more_whitespace) {
432 if (cp[str_index + 1] !=
'\0' && (cp[str_index + 1] <
'0' || cp[str_index + 1] >
'9')) {
437 *(va_arg(args,
int*)) = dummy.Atoi();
445 while (cp[str_index] >=
'0' && cp[str_index] <=
'9')
446 dummy += cp[str_index++];
447 *(va_arg(args,
int*)) = dummy.Atoi();
452 else if (fmt[fmt_index] ==
'*') {
460 if (fmt[fmt_index] != cp[str_index]) {
474 if (cp[str_index] !=
'\0')
505 if (!pattern.Contains(
"*"))
return this->Contains(pattern);
506 else if (pattern ==
"*")
return kTRUE;
508 std::unique_ptr<TObjArray> tok(pattern.Tokenize(
"*"));
509 Int_t n_tok = tok->GetEntries();
510 if (!pattern.BeginsWith(
"*"))
511 if (!BeginsWith(((TObjString*)tok->First())->GetString())) {
514 if (!pattern.EndsWith(
"*"))
515 if (!EndsWith(((TObjString*)tok->Last())->GetString())) {
519 Int_t idx = 0, num = 0;
520 for (Int_t ii = 0; ii < n_tok; ii += 1) {
521 idx = Index(((TObjString*)tok->At(ii))->GetString(), idx);
528 if (num == n_tok)
return kTRUE;
602 kObjArr.reset(Tokenize(delim));
735 if (strip_whitespace) st.Remove(kBoth,
' ');
803 kObjArr.reset(Tokenize(delim));
874 if (strip_whitespace) st.Remove(kBoth,
' ');
896 while (!copy.
End()) {
914 std::vector<KVString> v;
917 v.push_back(
Next(strip_whitespace));
922 #ifdef __WITH_KVSTRING_ITOA
929 Bool_t KVString::IsBin()
const
935 const char* cp = Data();
936 Ssiz_t len = Length();
937 if (len == 0)
return kFALSE;
938 for (Ssiz_t i = 0; i < len; ++i)
939 if (cp[i] !=
'0' && cp[i] !=
'1')
951 Bool_t KVString::IsOct()
const
957 const char* cp = Data();
958 Ssiz_t len = Length();
959 if (len == 0)
return kFALSE;
960 for (Ssiz_t i = 0; i < len; ++i)
961 if (!isdigit(cp[i]) || cp[i] ==
'8' || cp[i] ==
'9')
973 Bool_t KVString::IsDec()
const
979 const char* cp = Data();
980 Ssiz_t len = Length();
981 if (len == 0)
return kFALSE;
982 for (Ssiz_t i = 0; i < len; ++i)
996 Bool_t KVString::IsInBaseN(Int_t base)
const
1003 if (base < 2 || base > 36) {
1004 Error(
"KVString::IsInBaseN",
"base %d is not supported. Suppported bases are {2,3,...,36}.", base);
1007 if (Length() == 0) {
1008 Error(
"KVString::IsInBaseN",
"input string is empty.") ;
1013 KVString str_ref0 =
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ" ;
1015 str_ref.Remove(base) ;
1016 Bool_t isInBase = kTRUE ;
1017 for (Int_t k = 0; k < str.Length(); k++) {
1018 if (! str_ref.Contains(str[k])) {
1049 #ifdef __WITH_KVSTRING_ITOA
1052 if (base < 2 || base > 36) {
1053 Error(
"KVString::Itoa",
"base %d is not supported. Suppported bases are {2,3,...,36}.", base) ;
1057 Int_t quotient = value;
1060 buf +=
"0123456789abcdefghijklmnopqrstuvwxyz"[ TMath::Abs(quotient % base) ];
1065 if (value < 0) buf +=
'-';
1066 std::reverse(buf.begin(), buf.end());
1069 return TString::Itoa(value, base);
1088 #ifdef __WITH_KVSTRING_ITOA
1091 if (base < 2 || base > 36) {
1092 Error(
"KVString::UItoa",
"base %d is not supported. Suppported bases are {2,3,...,36}.", base);
1096 UInt_t quotient = value;
1099 buf +=
"0123456789abcdefghijklmnopqrstuvwxyz"[ quotient % base ];
1103 std::reverse(buf.begin(), buf.end());
1106 return TString::UItoa(value, base);
1125 #ifdef __WITH_KVSTRING_ITOA
1128 if (base < 2 || base > 36) {
1129 Error(
"KVString::LLtoa",
"base %d is not supported. Suppported bases are {2,3,...,36}.", base);
1133 Long64_t quotient = value;
1136 buf +=
"0123456789abcdefghijklmnopqrstuvwxyz"[ TMath::Abs(quotient % base) ];
1141 if (value < 0) buf +=
'-';
1142 std::reverse(buf.begin(), buf.end());
1145 return TString::LLtoa(value, base);
1164 #ifdef __WITH_KVSTRING_ITOA
1167 if (base < 2 || base > 36) {
1168 Error(
"KVString::ULLtoa",
"base %d is not supported. Suppported bases are {2,3,...,36}.", base);
1172 ULong64_t quotient = value;
1175 buf +=
"0123456789abcdefghijklmnopqrstuvwxyz"[ quotient % base ];
1179 std::reverse(buf.begin(), buf.end());
1182 return TString::ULLtoa(value, base);
1197 #ifdef __WITH_KVSTRING_ITOA
1200 if (base_in < 2 || base_in > 36 || base_out < 2 || base_out > 36) {
1201 Error(
"KVString::BaseConvert",
"only bases 2-36 are supported (base_in=%d, base_out=%d).", base_in, base_out);
1206 Bool_t isSigned = kFALSE;
1207 if (s_in_[0] ==
'-') {
1211 if (!isSigned && s_in_[0] ==
'+') s_in_.Remove(0, 1);
1212 if (base_in == 16 && s_in_.BeginsWith(
"0x")) s_in_.Remove(0, 2);
1213 s_in_ =
KVString(s_in_.Strip(KVString::kLeading,
'0'));
1215 if (!s_in_.IsInBaseN(base_in)) {
1216 Error(
"KVString::BaseConvert",
"s_in=\"%s\" is not in base %d", s_in.Data(), base_in);
1221 if (s_in_.Length() > s_max.Length()) {
1223 Error(
"KVString::BaseConvert",
"s_in=\"%s\" > %s = 2^64-1 in base %d.", s_in.Data(), s_max.Data(), base_in);
1226 else if (s_in_.Length() == s_max.Length()) {
1229 if (s_in_ > s_max) {
1231 Error(
"KVString::BaseConvert",
"s_in=\"%s\" > %s = 2^64-1 in base %d.", s_in.Data(), s_max.Data(), base_in);
1237 ULong64_t i = ULong64_t(strtoull(s_in.Data(), 0, base_in));
1239 if (isSigned) s_out.Prepend(
"-");
1242 return TString::BaseConvert(s_in, base_in, base_out);
1262 if (tmp.Length()) tmp +=
" ";
1284 while (!tmp.
End()) {
1285 if (tmp2.Length()) tmp2 +=
" ";
1326 while ((o = next())) {
1331 int tmplen = tmp.Length();
1333 int tmp2len = tmp2.Length();
1334 int len = TMath::Min(tmplen, tmp2len);
1335 for (
int i = 0; i < len; i++) {
1336 if (tmp[i] != tmp2[i]) tmp[i] = bug;
1338 if (tmp2len > tmplen) {
1339 tmp.Append(bug, tmp2len - tmplen);
1343 int tmplen = tmp.Length();
1345 bool do_bug =
false;
1346 for (
int i = 0; i < tmplen; i++) {
1348 if (tmp[i] == bug)
continue;
1349 else do_bug =
false;
1351 else if (tmp[i] == bug) {
1396 while ((o = next())) {
1398 tmp = o->GetTitle();
1401 int tmplen = tmp.Length();
1403 int tmp2len = tmp2.Length();
1404 int len = TMath::Min(tmplen, tmp2len);
1405 for (
int i = 0; i < len; i++) {
1406 if (tmp[i] != tmp2[i]) tmp[i] = bug;
1408 if (tmp2len > tmplen) {
1409 tmp.Append(bug, tmp2len - tmplen);
1413 int tmplen = tmp.Length();
1415 bool do_bug =
false;
1416 for (
int i = 0; i < tmplen; i++) {
1418 if (tmp[i] == bug)
continue;
1419 else do_bug =
false;
1421 else if (tmp[i] == bug) {
1442 for (Int_t i = 0; i < length; ++i) {
1443 UInt_t p = gRandom->Integer(52);
1444 if (p < 26) Append((
char)p +
'A');
1445 else Append((
char)(p - 26) +
'a');
1458 if ((*
this)[0] >=
'a' && (*
this)[0] <=
'z')(*this)[0] -= (
'a' -
'A');
1469 Double_t ey = error;
1471 TString sy = Format(
"%1.2e", y);
1472 TString sey = Format(
"%1.1e", ey);
1474 TString sy_dec, sy_exp, sey_dec, sey_exp;
1475 Double_t y_dec, ey_dec;
1476 Int_t y_exp, ey_exp;
1479 std::unique_ptr<TObjArray> loa_y(sy.Tokenize(
"e"));
1481 TIter next_y(loa_y.get());
1482 TObjString* os_y = 0;
1483 os_y = (TObjString*)next_y();
1484 sy_dec = os_y->GetString();
1485 os_y = (TObjString*)next_y();
1486 sy_exp = os_y->GetString();
1488 y_dec = sy_dec.Atof();
1489 y_exp = sy_exp.Atoi();
1492 std::unique_ptr<TObjArray> loa_ey(sey.Tokenize(
"e"));
1494 TIter next_ey(loa_ey.get());
1495 TObjString* os_ey = 0;
1497 os_ey = (TObjString*)next_ey();
1498 sey_dec = os_ey->GetString();
1500 os_ey = (TObjString*)next_ey();
1501 sey_exp = os_ey->GetString();
1503 ey_dec = sey_dec.Atof();
1504 ey_exp = sey_exp.Atoi();
1506 Double_t err = ey_dec * TMath::Power(10., ey_exp - y_exp);
1509 if (!((TString)Format(
"%1.2g", y_dec)).Contains(
".") && err >= 1) {
1511 if (!((TString)Format(
"%1.2g", err)).Contains(
".")) {
1512 if (y_exp == ey_exp) s = Format(
"%1.2g.0(%g.0).10$^{%d}$", y_dec, ey_dec, y_exp);
1513 else s = Format(
"%1.3g.0(%g.0).10$^{%d}$", y_dec, err, y_exp);
1515 else if (((TString)Format(
"%1.2g", err)) == ((TString)Format(
"%1.1g", err)) && ((TString)Format(
"%1.2g", err)).Contains(
".")) {
1516 if (y_exp == ey_exp) s = Format(
"%1.2g.0(%g0).10$^{%d}$", y_dec, ey_dec, y_exp);
1517 else s = Format(
"%1.3g.0(%g0).10$^{%d}$", y_dec, err, y_exp);
1520 if (y_exp == ey_exp) s = Format(
"%1.2g.0(%g).10$^{%d}$", y_dec, ey_dec, y_exp);
1521 else s = Format(
"%1.3g.0(%g).10$^{%d}$", y_dec, err, y_exp);
1524 else if (((TString)Format(
"%1.3g", y_dec)) == ((TString)Format(
"%1.2g", y_dec)) && ((TString)Format(
"%1.2g", y_dec)).Contains(
".") && err < 1) {
1526 if (!((TString)Format(
"%1.2g", err)).Contains(
".")) {
1527 if (y_exp == ey_exp) s = Format(
"%1.2g0(%g.0).10$^{%d}$", y_dec, ey_dec, y_exp);
1528 else s = Format(
"%1.3g0(%g.0).10$^{%d}$", y_dec, err, y_exp);
1530 else if (((TString)Format(
"%1.2g", err)) == ((TString)Format(
"%1.1g", err)) && ((TString)Format(
"%1.2g", err)).Contains(
".")) {
1531 if (y_exp == ey_exp) s = Format(
"%1.2g0(%g0).10$^{%d}$", y_dec, ey_dec, y_exp);
1532 else s = Format(
"%1.3g0(%g0).10$^{%d}$", y_dec, err, y_exp);
1535 if (y_exp == ey_exp) s = Format(
"%1.2g0(%g).10$^{%d}$", y_dec, ey_dec, y_exp);
1536 else s = Format(
"%1.3g0(%g).10$^{%d}$", y_dec, err, y_exp);
1539 else if (!((TString)Format(
"%1.2g", err)).Contains(
".")) {
1540 if (y_exp == ey_exp) s = Format(
"%1.2g(%g.0).10$^{%d}$", y_dec, ey_dec, y_exp);
1541 else s = Format(
"%1.3g(%g.0).10$^{%d}$", y_dec, err, y_exp);
1543 else if (((TString)Format(
"%1.2g", err)) == ((TString)Format(
"%1.1g", err)) && ((TString)Format(
"%1.2g", err)).Contains(
".")) {
1544 if (y_exp == ey_exp) s = Format(
"%1.2g(%g0).10$^{%d}$", y_dec, ey_dec, y_exp);
1545 else s = Format(
"%1.3g(%g0).10$^{%d}$", y_dec, err, y_exp);
1548 if (y_exp == ey_exp) s = Format(
"%1.2g(%g).10$^{%d}$", y_dec, ey_dec, y_exp);
1549 else s = Format(
"%1.3g(%g).10$^{%d}$", y_dec, err, y_exp);;
1552 s.ReplaceAll(
".10$^{0}$",
"");
1553 s.ReplaceAll(
"0)",
")");
1555 Form(
"%s", s.Data());
Extension of ROOT TString class which allows backwards compatibility with ROOT v3....
void Begin(TString delim) const
std::unique_ptr< TObjArray > kObjArr
used by Next() to iterate over list
void RBegin(TString delim) const
void RemoveAllExtraWhiteSpace()
KVString Next(Bool_t strip_whitespace=kFALSE) const
virtual Int_t Sscanf(const Char_t *fmt,...)
static KVString ULLtoa(ULong64_t value, Int_t base)
static KVString LLtoa(Long64_t value, Int_t base)
Bool_t fEndList
used by Next() & End() to iterate over list
void RandomLetterSequence(Int_t length)
std::vector< KVString > Vectorize(TString delim, Bool_t strip_whitespace=kFALSE)
virtual KVString & Substitute(const Char_t c1, const Char_t c2)
Replace every occurence of 'c1' with 'c2'.
KVString & FindCommonCharacters(const TCollection *, const char bug=' *')
static KVString BaseConvert(const KVString &s_in, Int_t base_in, Int_t base_out)
static KVString UItoa(UInt_t value, Int_t base)
virtual Bool_t Match(TString pattern)
Int_t GetNValues(TString delim) const
static KVString Itoa(Int_t value, Int_t base)
KVString RNext(Bool_t strip_whitespace=kFALSE) const
Int_t fIterIndex
used by Next() to iterate over list
KVString & FindCommonTitleCharacters(const TCollection *, const char bug=' *')
KVString StripAllExtraWhiteSpace() const
void Capitalize()
Change first character of string from lower to upper case.