1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
|
#pragma once
namespace nall {
//nall::format is a vector<string> of parameters that can be applied to a string
//each {#} token will be replaced with its appropriate format parameter
inline auto string::format(const nall::string_format& params) -> type& {
auto size = (s32)this->size();
auto data = memory::allocate<char>(size);
memory::copy(data, this->data(), size);
s32 x = 0;
while(x < size - 2) { //2 = minimum tag length
if(data[x] != '{') { x++; continue; }
s32 y = x + 1;
while(y < size - 1) { //-1 avoids going out of bounds on test after this loop
if(data[y] != '}') { y++; continue; }
break;
}
if(data[y++] != '}') { x++; continue; }
static auto isNumeric = [](char* s, char* e) -> bool {
if(s == e) return false; //ignore empty tags: {}
while(s < e) {
if(*s >= '0' && *s <= '9') { s++; continue; }
return false;
}
return true;
};
if(!isNumeric(&data[x + 1], &data[y - 1])) { x++; continue; }
u32 index = toNatural(&data[x + 1]);
if(index >= params.size()) { x++; continue; }
u32 sourceSize = y - x;
u32 targetSize = params[index].size();
u32 remaining = size - x;
if(sourceSize > targetSize) {
u32 difference = sourceSize - targetSize;
memory::move(&data[x], &data[x + difference], remaining - difference);
size -= difference;
} else if(targetSize > sourceSize) {
u32 difference = targetSize - sourceSize;
data = (char*)realloc(data, size + difference);
size += difference;
memory::move(&data[x + difference], &data[x], remaining);
}
memory::copy(&data[x], params[index].data(), targetSize);
x += targetSize;
}
resize(size);
memory::copy(get(), data, size);
memory::free(data);
return *this;
}
template<typename T, typename... P> inline auto string_format::append(const T& value, P&&... p) -> string_format& {
vector<string>::append(value);
return append(forward<P>(p)...);
}
inline auto string_format::append() -> string_format& {
return *this;
}
template<typename... P> inline auto print(P&&... p) -> void {
string s{forward<P>(p)...};
fwrite(s.data(), 1, s.size(), stdout);
fflush(stdout);
}
template<typename... P> inline auto print(FILE* fp, P&&... p) -> void {
string s{forward<P>(p)...};
fwrite(s.data(), 1, s.size(), fp);
if(fp == stdout || fp == stderr) fflush(fp);
}
template<typename T> inline auto pad(const T& value, long precision, char padchar) -> string {
string buffer{value};
if(precision) buffer.size(precision, padchar);
return buffer;
}
template<typename T> inline auto hex(T value, long precision, char padchar) -> string {
string buffer;
buffer.resize(sizeof(T) * 2);
char* p = buffer.get();
//create a mask to clear the upper four bits after shifting right in case T is a signed type
T mask = 1;
mask <<= sizeof(T) * 8 - 4;
mask -= 1;
u32 size = 0;
do {
u32 n = value & 15;
p[size++] = n < 10 ? '0' + n : 'a' + n - 10;
value = value >> 4 & mask;
} while(value);
buffer.resize(size);
buffer.reverse();
if(precision) buffer.size(precision, padchar);
return buffer;
}
template<typename T> inline auto octal(T value, long precision, char padchar) -> string {
string buffer;
buffer.resize(sizeof(T) * 3);
char* p = buffer.get();
//create a mask to clear the upper three bits
T mask = 1;
mask <<= sizeof(T) * 8 - 3;
mask -= 1;
u32 size = 0;
do {
p[size++] = '0' + (value & 7);
value = value >> 3 & mask;
} while(value);
buffer.resize(size);
buffer.reverse();
if(precision) buffer.size(precision, padchar);
return buffer;
}
template<typename T> inline auto binary(T value, long precision, char padchar) -> string {
string buffer;
buffer.resize(sizeof(T) * 8);
char* p = buffer.get();
//create a mask to clear the upper one bit
T mask = 1;
mask <<= sizeof(T) * 8 - 1;
mask -= 1;
u32 size = 0;
do {
p[size++] = '0' + (value & 1);
value = value >> 1 & mask;
} while(value);
buffer.resize(size);
buffer.reverse();
if(precision) buffer.size(precision, padchar);
return buffer;
}
}
|