// ************************************************************************** // * This file is part of the FreeFileSync project. It is distributed under * // * GNU General Public License: http://www.gnu.org/licenses/gpl.html * // * Copyright (C) Zenju (zenju AT gmx DOT de) - All Rights Reserved * // ************************************************************************** #ifndef STRING_TRAITS_HEADER_813274321443234 #define STRING_TRAITS_HEADER_813274321443234 #include "type_tools.h" //uniform access to string-like types, both classes and character arrays namespace zen { /* IsStringLike<>::value: IsStringLike::value; //equals "true" IsStringLike ::value; //equals "false" GetCharType<>::Type: GetCharType::Type //equals wchar_t GetCharType ::Type //equals wchar_t strLength(): strLength(str); //equals str.length() strLength(array); //equals cStringLength(array) strBegin(): -> not null-terminated! -> may be nullptr if length is 0! std::wstring str(L"dummy"); char array[] = "dummy"; strBegin(str); //returns str.c_str() strBegin(array); //returns array */ //reference a sub-string for consumption by zen string_tools template class StringRef { public: template StringRef(Iterator first, Iterator last) : length_(last - first), data_(first != last ? &*first : nullptr) {} size_t length() const { return length_; } const Char* data() const { return data_; } //1. no null-termination! 2. may be nullptr! private: size_t length_; const Char* data_; }; //---------------------- implementation ---------------------- namespace implementation { template //test if result of S::c_str() can convert to const Char* class HasConversion { typedef char Yes[1]; typedef char No [2]; static Yes& hasConversion(const Char*); static No& hasConversion(...); static S& createInstance(); public: enum { value = sizeof(hasConversion(createInstance().c_str())) == sizeof(Yes) }; }; template struct GetCharTypeImpl : ResultType {}; template struct GetCharTypeImpl : ResultType< typename SelectIf::value, wchar_t, typename SelectIf::value, char, NullType>::Type >::Type> { //typedef typename S::value_type Type; /*DON'T use S::value_type: 1. support Glib::ustring: value_type is "unsigned int" but c_str() returns "const char*" 2. wxString, wxWidgets v2.9, has some questionable string design: wxString::c_str() returns a proxy (wxCStrData) which is implicitly convertible to *both* "const char*" and "const wchar_t*" while wxString::value_type is a wrapper around an unsigned int */ }; template <> struct GetCharTypeImpl : ResultType {}; template <> struct GetCharTypeImpl : ResultType {}; ZEN_INIT_DETECT_MEMBER_TYPE(value_type); ZEN_INIT_DETECT_MEMBER(c_str); //we don't know the exact declaration of the member attribute and it may be in a base class! ZEN_INIT_DETECT_MEMBER(length); // template class StringTraits { typedef typename RemoveRef ::Type NonRefType; typedef typename RemoveConst ::Type NonConstType; typedef typename RemoveArray ::Type NonArrayType; typedef typename RemovePointer::Type NonPtrType; typedef typename RemoveConst ::Type UndecoratedType; //handle "const char* const" public: enum { isStringClass = HasMemberType_value_type::value && HasMember_c_str ::value && HasMember_length ::value }; typedef typename GetCharTypeImpl::Type CharType; enum { isStringLike = IsSameType::value || IsSameType::value }; }; template <> class StringTraits> { public: enum { isStringClass = false, isStringLike = true }; typedef char CharType; }; template <> class StringTraits> { public: enum { isStringClass = false, isStringLike = true }; typedef wchar_t CharType; }; } template struct IsStringLike : StaticBool::isStringLike> {}; template struct GetCharType : ResultType::CharType> {}; namespace implementation { template inline size_t cStringLength(const C* str) //naive implementation seems somewhat faster than "optimized" strlen/wcslen! { #if defined _MSC_VER && _MSC_VER > 1800 static_assert(false, "strlen/wcslen are vectorized in VS14 CTP3 -> test again!"); #endif static_assert(IsSameType::value || IsSameType::value, ""); size_t len = 0; while (*str++ != 0) ++len; return len; } } template inline const typename GetCharType::Type* strBegin(const S& str, typename EnableIf::isStringClass>::Type* = nullptr) //SFINAE: T must be a "string" { return str.c_str(); } inline const char* strBegin(const char* str) { return str; } inline const wchar_t* strBegin(const wchar_t* str) { return str; } inline const char* strBegin(const char& ch) { return &ch; } inline const wchar_t* strBegin(const wchar_t& ch) { return &ch; } inline const char* strBegin(const StringRef& ref) { return ref.data(); } inline const wchar_t* strBegin(const StringRef& ref) { return ref.data(); } template inline size_t strLength(const S& str, typename EnableIf::isStringClass>::Type* = nullptr) //SFINAE: T must be a "string" { return str.length(); } inline size_t strLength(const char* str) { return implementation::cStringLength(str); } inline size_t strLength(const wchar_t* str) { return implementation::cStringLength(str); } inline size_t strLength(char) { return 1; } inline size_t strLength(wchar_t) { return 1; } inline size_t strLength(const StringRef& ref) { return ref.length(); } inline size_t strLength(const StringRef& ref) { return ref.length(); } } #endif //STRING_TRAITS_HEADER_813274321443234