mirror of
https://github.com/bolero-MURAKAMI/Sprout.git
synced 2024-11-29 01:03:45 +00:00
159 lines
5.7 KiB
C++
159 lines
5.7 KiB
C++
/*=============================================================================
|
|
Copyright (c) 2011-2019 Bolero MURAKAMI
|
|
https://github.com/bolero-MURAKAMI/Sprout
|
|
|
|
Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
=============================================================================*/
|
|
#ifndef SPROUT_STRING_CONCAT_HPP
|
|
#define SPROUT_STRING_CONCAT_HPP
|
|
|
|
#include <type_traits>
|
|
#include <sprout/config.hpp>
|
|
#include <sprout/workaround/std/cstddef.hpp>
|
|
#include <sprout/index_tuple/metafunction.hpp>
|
|
#include <sprout/string/char_traits.hpp>
|
|
#include <sprout/string/string.hpp>
|
|
|
|
namespace sprout {
|
|
namespace detail {
|
|
template<typename T, std::size_t N, typename Traits, sprout::index_t... Indexes>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N + 1, Traits>
|
|
string_concat(
|
|
sprout::basic_string<T, N, Traits> const& lhs, std::size_t lsize,
|
|
T const& rhs,
|
|
sprout::index_tuple<Indexes...>
|
|
)
|
|
{
|
|
typedef sprout::detail::string_construct_access<T, N + 1, Traits> access_type;
|
|
return access_type::raw_construct(
|
|
lsize + 1,
|
|
(Indexes < lsize ? lhs[Indexes]
|
|
: Indexes < lsize + 1 ? rhs
|
|
: T()
|
|
)...
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits, sprout::index_t... Indexes>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, 1 + N, Traits>
|
|
string_concat(
|
|
T const& lhs,
|
|
sprout::basic_string<T, N, Traits> const& rhs, std::size_t rsize,
|
|
sprout::index_tuple<Indexes...>
|
|
)
|
|
{
|
|
typedef sprout::detail::string_construct_access<T, 1 + N, Traits> access_type;
|
|
return access_type::raw_construct(
|
|
1 + rsize,
|
|
(Indexes < 1 ? lhs
|
|
: Indexes < 1 + rsize ? rhs[Indexes - 1]
|
|
: T()
|
|
)...
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits, std::size_t M, sprout::index_t... Indexes>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N + (M - 1), Traits>
|
|
string_concat(
|
|
sprout::basic_string<T, N, Traits> const& lhs, std::size_t lsize,
|
|
T const (& rhs)[M], std::size_t rsize,
|
|
sprout::index_tuple<Indexes...>
|
|
)
|
|
{
|
|
typedef sprout::detail::string_construct_access<T, N + (M - 1), Traits> access_type;
|
|
return access_type::raw_construct(
|
|
lsize + rsize,
|
|
(Indexes < lsize ? lhs[Indexes]
|
|
: Indexes < lsize + rsize ? rhs[Indexes - lsize]
|
|
: T()
|
|
)...
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits, std::size_t M, sprout::index_t... Indexes>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, (M - 1) + N, Traits>
|
|
string_concat(
|
|
T const (& lhs)[M], std::size_t lsize,
|
|
sprout::basic_string<T, N, Traits> const& rhs, std::size_t rsize,
|
|
sprout::index_tuple<Indexes...>
|
|
)
|
|
{
|
|
typedef sprout::detail::string_construct_access<T, (M - 1) + N, Traits> access_type;
|
|
return access_type::raw_construct(
|
|
lsize + rsize,
|
|
(Indexes < lsize ? *(lhs + Indexes)
|
|
: Indexes < lsize + rsize ? rhs[Indexes - lsize]
|
|
: T()
|
|
)...
|
|
);
|
|
}
|
|
template<typename T, std::size_t N1, std::size_t N2, typename Traits, sprout::index_t... Indexes>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N1 + N2, Traits>
|
|
string_concat(
|
|
sprout::basic_string<T, N1, Traits> const& lhs, std::size_t lsize,
|
|
sprout::basic_string<T, N2, Traits> const& rhs, std::size_t rsize,
|
|
sprout::index_tuple<Indexes...>
|
|
)
|
|
{
|
|
typedef sprout::detail::string_construct_access<T, N1 + N2, Traits> access_type;
|
|
return access_type::raw_construct(
|
|
lsize + rsize,
|
|
(Indexes < lsize ? lhs[Indexes]
|
|
: Indexes < lsize + rsize ? rhs[Indexes - lsize]
|
|
: T()
|
|
)...
|
|
);
|
|
}
|
|
} // namespace detail
|
|
|
|
//
|
|
// operator+
|
|
//
|
|
template<typename T, std::size_t N, typename Traits>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N + 1, Traits>
|
|
operator+(sprout::basic_string<T, N, Traits> const& lhs, T const& rhs) {
|
|
return sprout::detail::string_concat(
|
|
lhs, lhs.size(),
|
|
rhs,
|
|
sprout::make_index_tuple<N + 1>::make()
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, 1 + N, Traits>
|
|
operator+(T const& lhs, sprout::basic_string<T, N, Traits> const& rhs) {
|
|
return sprout::detail::string_concat(
|
|
lhs,
|
|
rhs, rhs.size(),
|
|
sprout::make_index_tuple<1 + N>::make()
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits, std::size_t M>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N + (M - 1), Traits>
|
|
operator+(sprout::basic_string<T, N, Traits> const& lhs, T const (& rhs)[M]) {
|
|
typedef sprout::char_traits_helper<Traits> traits_type;
|
|
return sprout::detail::string_concat(
|
|
lhs, lhs.size(),
|
|
rhs, traits_type::length(rhs, M - 1),
|
|
sprout::make_index_tuple<N + (M - 1)>::make()
|
|
);
|
|
}
|
|
template<typename T, std::size_t N, typename Traits, std::size_t M>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, (M - 1) + N, Traits>
|
|
operator+(T const (& lhs)[M], sprout::basic_string<T, N, Traits> const& rhs) {
|
|
typedef sprout::char_traits_helper<Traits> traits_type;
|
|
return sprout::detail::string_concat(
|
|
lhs, traits_type::length(lhs, M - 1),
|
|
rhs, rhs.size(),
|
|
sprout::make_index_tuple<(M - 1) + N>::make()
|
|
);
|
|
}
|
|
template<typename T, std::size_t N1, std::size_t N2, typename Traits>
|
|
inline SPROUT_CONSTEXPR sprout::basic_string<T, N1 + N2, Traits>
|
|
operator+(sprout::basic_string<T, N1, Traits> const& lhs, sprout::basic_string<T, N2, Traits> const& rhs) {
|
|
return sprout::detail::string_concat(
|
|
lhs, lhs.size(),
|
|
rhs, rhs.size(),
|
|
sprout::make_index_tuple<N1 + N2>::make()
|
|
);
|
|
}
|
|
} // namespace sprout
|
|
|
|
#endif // #ifndef SPROUT_STRING_CONCAT_HPP
|