17 #elif (XTD_HAS_EXP_CODECVT)
18 #include <experimental/codecvt>
38 template <
typename,
typename>
class xstring_format;
46 template <
typename _ChT>
class xstring :
public std::basic_string<_ChT>{
48 using _super_t = std::basic_string<_ChT>;
49 using size_type =
typename _super_t::size_type;
52 template <
typename ... _ArgsT>
54 : _super_t(std::forward<_ArgsT>(oArgs)...){}
63 xstring sRet = ((_ChT*)
"\0\0\0");
68 template <
typename _Ty,
typename ... _ArgsT>
70 xstring sRet = _::xstring_format<_ChT, const _Ty &>::format(static_cast<const _Ty &>(val));
71 sRet +=
format(std::forward<_ArgsT>(oArgs)...);
81 for (
auto & ch : *
this){
82 ch = std::tolower(ch, loc);
92 for (
auto & ch : *
this){
93 ch = std::toupper(ch, loc);
100 auto oBegin = _super_t::begin();
101 if (oBegin >= _super_t::end()){
104 for(;oBegin != _super_t::end() ; ++oBegin){
105 if (!std::iswspace(*oBegin)){
109 _super_t::erase(_super_t::begin(), oBegin);
115 auto oBegin = _super_t::end();
116 if (oBegin <= _super_t::begin()){
120 for(;oBegin != _super_t::begin() ; --oBegin){
121 if (!std::iswspace(*oBegin) && *oBegin){
125 _super_t::erase(++oBegin, _super_t::end());
136 xstring& replace(std::initializer_list<_ChT> oItems, _ChT chReplace){
137 for (
auto & oCh : *
this){
139 for (
const auto & oFind : oItems){
153 size_type
find_first_of(
const std::initializer_list<_ChT>& delimiters, size_type pos = 0)
const{
154 size_type sRet = _super_t::npos;
155 for (
const _ChT ch : delimiters){
156 auto x = _super_t::find_first_of(ch, pos);
157 if ((_super_t::npos != x) && (_super_t::npos == sRet || x < sRet)){
165 std::vector<xstring<_ChT>>
split(
const std::initializer_list<_ChT>& delimiters,
bool trimEmpty =
false)
const{
166 using container_t = std::vector<xstring<_ChT>>;
170 size_type lastPos = 0;
172 using value_type =
typename container_t::value_type;
176 if (pos == _my_t::npos){
177 pos = _super_t::length();
178 if (pos != lastPos || !trimEmpty){
179 oRet.push_back(value_type(_super_t::data() + lastPos, (pos - lastPos)));
182 }
else if (pos != lastPos || !trimEmpty){
183 oRet.push_back(value_type(_super_t::data() + lastPos , (pos - lastPos)));
191 #if (!(XTD_HAS_CODECVT | XTD_HAS_EXP_CODECVT)) && (XTD_HAS_ICONV)
196 iconv_helper(
const iconv_helper&) =
delete;
197 iconv_helper& operator=(
const iconv_helper&) =
delete;
198 iconv_helper(
const char * to,
const char * from) : _iconv(iconv_open(to, from)){
199 if ((iconv_t)-1 == _iconv){
200 throw std::runtime_error(
"Failure creating iconv handle");
206 operator iconv_t(){
return _iconv; }
216 #if (XTD_HAS_CODECVT || XTD_HAS_EXP_CODECVT)
217 template <>
class xstring_format<char, const wchar_t *
const &>{
219 static inline string format(
const wchar_t *
const & src){
220 std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>,
wchar_t> oConv;
221 return oConv.to_bytes(src);
225 template <>
class xstring_format<wchar_t, const char *
const &>{
227 static inline wstring format(
const char *
const & src){
228 std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>,
wchar_t> oConv;
229 return oConv.from_bytes(src);
233 #elif (XTD_HAS_ICONV)
235 template <>
class xstring_format<char, const wchar_t *
const &>{
237 static string format(
const wchar_t *
const & src){
238 static iconv_helper oIconv(
"UTF-8",
"WCHAR_T");
239 size_t srclen = wcslen(src);
240 string sRet(srclen, 0);
241 size_t retlen = srclen;
242 iconv(oIconv,
nullptr,
nullptr,
nullptr, &retlen);
244 retlen = sRet.size();
245 auto pSrc =
reinterpret_cast<const char *
>(src);
246 auto iSrcLen = srclen *
sizeof(wchar_t);
247 auto sRetPtr = &sRet[0];
248 auto iRet = iconv(oIconv, const_cast<char**>(&pSrc), &iSrcLen, &sRetPtr, &retlen);
249 if (static_cast<size_t>(-1) != iRet){
253 throw std::runtime_error(
"A string conversion error occurred");
255 sRet.resize(sRet.size() * 2);
260 template <>
class xstring_format<wchar_t, const char *
const &>{
262 static inline wstring format(
const char *
const & src){
263 static iconv_helper oIconv(
"WCHAR_T",
"UTF-8");
264 size_t srclen = strlen(src);
265 wstring sRet(srclen, 0);
266 size_t retlen = srclen;
267 iconv(oIconv,
nullptr,
nullptr,
nullptr, &retlen);
269 retlen = sRet.size() *
sizeof(wchar_t);
270 auto pSrc =
const_cast<char*
>(src);
271 auto iSrcLen = srclen;
272 auto sRetPtr =
reinterpret_cast<char*
>(&sRet[0]);
273 auto iRet = iconv(oIconv, &pSrc, &iSrcLen, &sRetPtr, &retlen);
274 if (static_cast<size_t>(-1) != iRet){
278 throw std::runtime_error(
"A string conversion error occurred");
280 sRet.resize(sRet.size() * 2);
286 template <>
class xstring_format<char, const wchar_t *
const &>{
288 static string format(
const wchar_t *
const & src){
289 size_t srclen = wcslen(src);
290 string sRet(srclen, 0);
292 srclen = wcstombs(&sRet[0], src, sRet.size());
293 if (static_cast<size_t>(-1) == srclen){
294 throw std::runtime_error(
"A string conversion error occurred");
296 if (srclen < sRet.size()){
299 sRet.resize(srclen * 2);
306 template <>
class xstring_format<wchar_t, const char *
const &>{
309 static inline wstring format(
const char *
const & src){
310 size_t srclen = strlen(src);
311 wstring sRet(1+srclen,
'\0');
313 srclen = mbstowcs(&sRet[0], src, sRet.size());
314 if (static_cast<size_t>(-1) == srclen){
315 throw std::runtime_error(
"A string conversion error occurred");
317 if (srclen <= sRet.size()){
320 sRet.resize(srclen * 2);
329 template <
typename _ChT>
class xstring_format<_ChT, const _ChT *
const &>{
331 inline static xstring<_ChT> format(
const _ChT *
const & src){
332 return xstring<_ChT>(src);
336 template <
typename _ChT,
typename _Ch2,
size_t _Len>
class xstring_format<_ChT, const _Ch2(&)[_Len]>{
338 inline static xstring<_ChT> format(
const _Ch2(&src)[_Len]){
339 return xstring_format<_ChT, const _Ch2 * const &>::format(src);
343 template <
typename _ChT,
typename _Ch2>
class xstring_format<_ChT, const xtd::xstring<_Ch2> &>{
349 template <
typename _ChT,
typename _Ch2>
class xstring_format<_ChT, const std::basic_string<_Ch2> &>{
351 inline static xstring<_ChT> format(
const std::basic_string<_Ch2> & src){
357 template <>
class xstring_format<char, const void *
const &>{
359 inline static string format(
const void *
const & value){
360 return std::to_string(reinterpret_cast<size_t>(value));
365 template <>
class xstring_format<char, const int32_t &>{
367 inline static string format(
const int32_t & value){
368 return std::to_string(value);
372 template <>
class xstring_format<char, const uint32_t &>{
374 inline static string format(
const uint32_t & value){
375 return std::to_string(value);
379 template <>
class xstring_format<char, const int64_t &>{
381 inline static string format(
const int64_t & value){
382 return std::to_string(value);
386 template <>
class xstring_format<char, const uint64_t &>{
388 inline static string format(
const uint64_t & value){
389 return std::to_string(value);
394 template <>
class xstring_format<char, void *
const &>{
396 inline static string format(
const void *
const & value){
397 return std::to_string(reinterpret_cast<size_t>(value));
401 #if ((XTD_OS_MINGW | XTD_OS_WINDOWS) & XTD_OS)
402 template <>
class xstring_format<char, const DWORD &>{
404 inline static string format(
const DWORD & value){
405 return std::to_string(value);
xstring & to_upper(const std::locale &loc)
Converts the string to upper case.
xstring & rtrim()
Trim trailing whitespace.
xstring & ltrim()
Trim leading whitespace.
xstring(_ArgsT &&...oArgs)
Generic constructor forwards everything to the base class.
std::vector< xstring< _ChT > > split(const std::initializer_list< _ChT > &delimiters, bool trimEmpty=false) const
splits the string by the specified delmiters into constituent elements
host, target and build configurations and settings Various components are purpose built for specific ...
xstring & to_lower(const std::locale &loc)
Converts the string to lower case.
Extends std::string with some added functionality.
size_type find_first_of(const std::initializer_list< _ChT > &delimiters, size_type pos=0) const
finds the first occurance of any item
static xstring format()
Type safe formatting Appends each item in the parameter list together performing type-safe verificati...