Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add 'boost::intersperse' string algorithm #111

Open
wants to merge 4 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
23 changes: 23 additions & 0 deletions include/boost/algorithm/string/detail/formatter.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
#include <boost/range/begin.hpp>
#include <boost/range/end.hpp>
#include <boost/range/const_iterator.hpp>
#include <boost/range/join.hpp>

#include <boost/algorithm/string/detail/util.hpp>

Expand Down Expand Up @@ -111,6 +112,28 @@ namespace boost {
FinderT m_Finder;
};

// joint format functor ----------------------------------------------------//

// joint format functor
template<typename RangeT>
struct join_formatF
{
public:
// Construction
join_formatF(const RangeT& Other) :
m_Other(Other) {}

template<typename Range2T>
inline joined_range<
const Range2T, const RangeT>
operator()(const Range2T& Replace) const
{
return ::boost::range::join(Replace, m_Other);
}

private:
RangeT m_Other;
};

} // namespace detail
} // namespace algorithm
Expand Down
56 changes: 56 additions & 0 deletions include/boost/algorithm/string/detail/intersperse.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
// Boost string_algo library intersperse.hpp header file ---------------------------//

// Copyright Denis Mikhailov 2022.
//
// 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)

// See http://www.boost.org/ for updates, documentation, and revision history.

#ifndef BOOST_STRING_INTERSPERSE_DETAIL_HPP
#define BOOST_STRING_INTERSPERSE_DETAIL_HPP

#include <boost/algorithm/string/config.hpp>
#include <iterator>

#include <boost/range/size_type.hpp>
#include <boost/range/value_type.hpp>
#include <boost/iterator/iterator_adaptor.hpp> // for iterator_adaptor

namespace boost {
namespace algorithm {
namespace detail {

// just workaround for https://github.com/boostorg/iterator/issues/75
// FIXME remove this when the issue will be resolved
template<typename It>
struct just_iterator_adaptor : iterator_adaptor<just_iterator_adaptor<It>, It> {
explicit just_iterator_adaptor(It p)
: just_iterator_adaptor::iterator_adaptor_(p) {}
};

// functor to be passed into ::boost::token_finder
template<typename SinglePassRange>
class non_last_condition {
public:
typedef BOOST_STRING_TYPENAME range_size<SinglePassRange>::type size_type;
typedef BOOST_STRING_TYPENAME range_value<SinglePassRange>::type value_type;

explicit non_last_condition(size_type s, size_type& counter)
: m_Size(s), m_Counter(counter) { }
bool operator() (const value_type&) const {
return ++m_Counter < m_Size;
}
private:
const size_type m_Size;
size_type& m_Counter;
};


} // namespace detail
} // namespace algorithm
} // namespace boost


#endif // BOOST_STRING_INTERSPERSE_DETAIL_HPP
21 changes: 21 additions & 0 deletions include/boost/algorithm/string/formatter.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,26 @@ namespace boost {
return detail::dissect_formatF<FinderT>(Finder);
}

//! Join formatter
/*!
Constructs a \c join_formatter. Join formatter uses an other range to concatenate parameter
with the range. The joining of parameter and the range is returned
as a result

\param Other a range used to be concatenated with parameter
\return An instance of the \c join_formatter object.
*/
template<typename RangeT>
inline detail::join_formatF<
iterator_range<
BOOST_STRING_TYPENAME range_const_iterator<RangeT>::type> >
join_formatter(const RangeT& Other)
{
return detail::join_formatF<
iterator_range<
BOOST_STRING_TYPENAME range_const_iterator<RangeT>::type> >(::boost::as_literal(Other));
}


} // namespace algorithm

Expand All @@ -112,6 +132,7 @@ namespace boost {
using algorithm::identity_formatter;
using algorithm::empty_formatter;
using algorithm::dissect_formatter;
using algorithm::join_formatter;

} // namespace boost

Expand Down
182 changes: 182 additions & 0 deletions include/boost/algorithm/string/intersperse.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,182 @@
// Boost string_algo library intersperse.hpp header file ---------------------------//

// Copyright Denis Mikhailov 2022.
//
// 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)

// See http://www.boost.org/ for updates, documentation, and revision history.

#ifndef BOOST_STRING_INTERSPERSE_HPP
#define BOOST_STRING_INTERSPERSE_HPP

#include <boost/algorithm/string/config.hpp>

#include <boost/range.hpp>
#include <boost/range/begin.hpp>
#include <boost/range/end.hpp>
#include <boost/range/size.hpp>
#include <boost/range/size_type.hpp>
#include <boost/range/iterator_range_core.hpp>
#include <boost/range/as_literal.hpp>
#include <boost/iterator/function_input_iterator.hpp>
#include <boost/core/addressof.hpp>

#include <boost/algorithm/string/detail/intersperse.hpp>
#include <boost/algorithm/string/find_format.hpp>
#include <boost/algorithm/string/formatter.hpp>
#include <boost/algorithm/string/finder.hpp>

/*! \file
Defines intersperse algorithms.
Intersperse algorithms are used to insert value between each contiguous elements
from a sequence (string). Also, variants with (\c _fill) postfix can insert a whole
range instead of value.

Parametric (\c _generate) variants use a generator (functor) to obtain characters
to be inserted..
Functions take a nullary function object as a parameter, which is used to extract
a new character and then insert.
*/

namespace boost {
namespace algorithm {

// intersperse fill -----------------------------------------------//


//! Intersperse fill
/*!
Insert string between each contiguous elements from the input.
The input sequence is modified in-place.

\param Input An input sequence
\param Fill A string to be inserted
*/
template<typename SequenceT, typename RangeT>
inline void intersperse_fill(SequenceT& Input, const RangeT& Fill)
{
BOOST_STRING_TYPENAME ::boost::range_size<SequenceT>::type counter = 0;
::boost::find_format_all(
Input,
::boost::token_finder(
detail::non_last_condition<SequenceT>(::boost::size(Input), counter)),
::boost::join_formatter(::boost::as_literal(Fill)));
}

//! Intersperse fill
/*!
Insert string between each contiguous elements from the input.
The result is an interspersed copy of the input

\param Input An input sequence
\param Fill A string to be inserted
\return An interspersed copy of the input

\note This function provides the strong exception-safety guarantee
*/
template<typename SequenceT, typename RangeT>
inline SequenceT intersperse_fill_copy(
const SequenceT& Input,
const RangeT& Fill)
{
BOOST_STRING_TYPENAME ::boost::range_size<SequenceT>::type counter = 0;
return
::boost::find_format_all_copy(
Input,
::boost::token_finder(
detail::non_last_condition<SequenceT>(::boost::size(Input), counter)),
::boost::join_formatter(::boost::as_literal(Fill)));
}

// intersperse -----------------------------------------------//

//! Intersperse - parametric
/*!
Insert generated char between each contiguous elements from the input.
The input sequence is modified in-place.

\param Input An input sequence
\param G A generator(nullary function object) used to create insertable char
*/
template<typename SequenceT, typename GeneratorT>
inline void intersperse_generate(SequenceT& Input, GeneratorT G)
{
typedef ::boost::function_input_iterator<GeneratorT, int> It;
intersperse_fill(Input, boost::make_iterator_range(
detail::just_iterator_adaptor<It>(::boost::make_function_input_iterator(G, 0)),
detail::just_iterator_adaptor<It>(::boost::make_function_input_iterator(G, 1))));
}

//! Intersperse - parametric
/*!
Insert generated char between each contiguous elements from the input.
The result is an interspersed copy of the input.
\param Input An input sequence
\param G A generator(nullary function object) used to create insertable char
\return An interspersed copy of the input
*/
template<typename SequenceT, typename GeneratorT>
inline SequenceT intersperse_generate_copy(
const SequenceT& Input,
GeneratorT G)
{
typedef ::boost::function_input_iterator<GeneratorT, int> It;
return
intersperse_fill_copy(Input, ::boost::make_iterator_range(
detail::just_iterator_adaptor<It>(::boost::make_function_input_iterator(G, 0)),
detail::just_iterator_adaptor<It>(::boost::make_function_input_iterator(G, 1))));
}

//! Intersperse
/*!
Insert char between each contiguous elements from the input.
The input sequence is modified in-place.

\param Input An input sequence
\param Value A char to be inserted
*/
template<typename SequenceT, typename T>
inline void intersperse(SequenceT& Input, const T& Value)
{
intersperse_fill(Input, ::boost::make_iterator_range(
::boost::addressof(Value),
::boost::addressof(Value)+1));
}

//! Intersperse
/*!
Insert char between each contiguous elements from the input.
The result is an interspersed copy of the input

\param Input An input sequence
\param Value A char to be inserted
\return An interspersed copy of the input

\note This function provides the strong exception-safety guarantee
*/
template<typename SequenceT, typename T>
inline SequenceT intersperse_copy(
const SequenceT& Input,
const T& Value)
{
return
intersperse_fill_copy(Input, ::boost::make_iterator_range(
::boost::addressof(Value),
::boost::addressof(Value)+1));
}

} // namespace algorithm

// pull names to the boost namespace
using algorithm::intersperse_fill;
using algorithm::intersperse_fill_copy;
using algorithm::intersperse_generate;
using algorithm::intersperse_generate_copy;
using algorithm::intersperse;
using algorithm::intersperse_copy;

} // namespace boost

#endif // BOOST_STRING_INTERSPERSE_HPP
6 changes: 6 additions & 0 deletions string/test/Jamfile.v2
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,12 @@ test-suite algorithm/string
:
: trim
]
[ run
intersperse_test.cpp unit_test_framework
: :
:
: intersperse
]
[ run
conv_test.cpp unit_test_framework
: :
Expand Down
Loading