2014-02-11 07:32:56 +01:00
|
|
|
/*
|
|
|
|
* Created by Phil on 18/4/2013.
|
|
|
|
* Copyright 2013 Two Blue Cubes Ltd. All rights reserved.
|
|
|
|
*
|
|
|
|
* 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)
|
|
|
|
*/
|
|
|
|
// Only use header guard if we are not using an outer namespace
|
|
|
|
#ifndef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE
|
|
|
|
# ifdef TWOBLUECUBES_TEXT_FORMAT_H_INCLUDED
|
|
|
|
# ifndef TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED
|
|
|
|
# define TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED
|
|
|
|
# endif
|
|
|
|
# else
|
|
|
|
# define TWOBLUECUBES_TEXT_FORMAT_H_INCLUDED
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
#ifndef TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
// Use optional outer namespace
|
|
|
|
#ifdef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE
|
|
|
|
namespace CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace Tbc {
|
|
|
|
|
|
|
|
#ifdef TBC_TEXT_FORMAT_CONSOLE_WIDTH
|
|
|
|
const unsigned int consoleWidth = TBC_TEXT_FORMAT_CONSOLE_WIDTH;
|
|
|
|
#else
|
|
|
|
const unsigned int consoleWidth = 80;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
struct TextAttributes {
|
|
|
|
TextAttributes()
|
|
|
|
: initialIndent( std::string::npos ),
|
|
|
|
indent( 0 ),
|
2017-01-17 18:13:23 +01:00
|
|
|
width( consoleWidth-1 )
|
2014-02-11 07:32:56 +01:00
|
|
|
{}
|
|
|
|
|
|
|
|
TextAttributes& setInitialIndent( std::size_t _value ) { initialIndent = _value; return *this; }
|
|
|
|
TextAttributes& setIndent( std::size_t _value ) { indent = _value; return *this; }
|
|
|
|
TextAttributes& setWidth( std::size_t _value ) { width = _value; return *this; }
|
|
|
|
|
|
|
|
std::size_t initialIndent; // indent of first line, or npos
|
|
|
|
std::size_t indent; // indent of subsequent lines, or all if initialIndent is npos
|
|
|
|
std::size_t width; // maximum width of text, including indent. Longer text will wrap
|
|
|
|
};
|
|
|
|
|
|
|
|
class Text {
|
|
|
|
public:
|
|
|
|
Text( std::string const& _str, TextAttributes const& _attr = TextAttributes() )
|
|
|
|
: attr( _attr )
|
2017-07-02 10:52:29 +02:00
|
|
|
{
|
|
|
|
init( _str );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
Text( T const& _val, TextAttributes const& _attr = TextAttributes() )
|
|
|
|
: attr( _attr )
|
|
|
|
{
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << _val;
|
|
|
|
init( oss.str() );
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef std::vector<std::string>::const_iterator const_iterator;
|
|
|
|
|
|
|
|
const_iterator begin() const { return lines.begin(); }
|
|
|
|
const_iterator end() const { return lines.end(); }
|
|
|
|
std::string const& last() const { return lines.back(); }
|
|
|
|
std::size_t size() const { return lines.size(); }
|
|
|
|
std::string const& operator[]( std::size_t _index ) const { return lines[_index]; }
|
|
|
|
std::string toString() const {
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << *this;
|
|
|
|
return oss.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline friend std::ostream& operator << ( std::ostream& _stream, Text const& _text ) {
|
|
|
|
for( Text::const_iterator it = _text.begin(), itEnd = _text.end();
|
|
|
|
it != itEnd; ++it ) {
|
|
|
|
if( it != _text.begin() )
|
|
|
|
_stream << "\n";
|
|
|
|
_stream << *it;
|
|
|
|
}
|
|
|
|
return _stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
TextAttributes attr;
|
|
|
|
std::vector<std::string> lines;
|
|
|
|
|
|
|
|
void init( std::string const& _str )
|
2014-02-11 07:32:56 +01:00
|
|
|
{
|
2017-01-17 18:13:23 +01:00
|
|
|
const std::string wrappableBeforeChars = "[({<\t";
|
|
|
|
const std::string wrappableAfterChars = "])}>-,./|\\";
|
|
|
|
const std::string wrappableInsteadOfChars = " \n\r";
|
2017-07-02 10:52:29 +02:00
|
|
|
std::string indent = attr.initialIndent != std::string::npos
|
|
|
|
? std::string( attr.initialIndent, ' ' )
|
|
|
|
: std::string( attr.indent, ' ' );
|
2017-01-17 18:13:23 +01:00
|
|
|
|
|
|
|
typedef std::string::const_iterator iterator;
|
|
|
|
iterator it = _str.begin();
|
|
|
|
const iterator strEnd = _str.end();
|
|
|
|
|
|
|
|
while( it != strEnd ) {
|
2014-02-11 07:32:56 +01:00
|
|
|
|
|
|
|
if( lines.size() >= 1000 ) {
|
|
|
|
lines.push_back( "... message truncated due to excessive size" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-17 18:13:23 +01:00
|
|
|
|
|
|
|
std::string suffix;
|
2017-07-02 10:52:29 +02:00
|
|
|
std::size_t width = (std::min)( static_cast<size_t>( strEnd-it ), attr.width-static_cast<size_t>( indent.size() ) );
|
2017-01-17 18:13:23 +01:00
|
|
|
iterator itEnd = it+width;
|
|
|
|
iterator itNext = _str.end();
|
|
|
|
|
|
|
|
iterator itNewLine = std::find( it, itEnd, '\n' );
|
|
|
|
if( itNewLine != itEnd )
|
|
|
|
itEnd = itNewLine;
|
|
|
|
|
|
|
|
if( itEnd != strEnd ) {
|
|
|
|
bool foundWrapPoint = false;
|
2017-01-23 17:58:49 +01:00
|
|
|
iterator findIt = itEnd;
|
|
|
|
do {
|
2017-01-17 18:13:23 +01:00
|
|
|
if( wrappableAfterChars.find( *findIt ) != std::string::npos && findIt != itEnd ) {
|
|
|
|
itEnd = findIt+1;
|
|
|
|
itNext = findIt+1;
|
|
|
|
foundWrapPoint = true;
|
|
|
|
}
|
|
|
|
else if( findIt > it && wrappableBeforeChars.find( *findIt ) != std::string::npos ) {
|
|
|
|
itEnd = findIt;
|
|
|
|
itNext = findIt;
|
|
|
|
foundWrapPoint = true;
|
|
|
|
}
|
|
|
|
else if( wrappableInsteadOfChars.find( *findIt ) != std::string::npos ) {
|
|
|
|
itNext = findIt+1;
|
|
|
|
itEnd = findIt;
|
|
|
|
foundWrapPoint = true;
|
|
|
|
}
|
2017-01-23 17:58:49 +01:00
|
|
|
if( findIt == it )
|
|
|
|
break;
|
|
|
|
else
|
|
|
|
--findIt;
|
2017-01-17 18:13:23 +01:00
|
|
|
}
|
2017-01-23 17:58:49 +01:00
|
|
|
while( !foundWrapPoint );
|
|
|
|
|
2017-01-17 18:13:23 +01:00
|
|
|
if( !foundWrapPoint ) {
|
|
|
|
// No good wrap char, so we'll break mid word and add a hyphen
|
|
|
|
--itEnd;
|
|
|
|
itNext = itEnd;
|
|
|
|
suffix = "-";
|
2014-02-11 07:32:56 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-01-17 18:13:23 +01:00
|
|
|
while( itEnd > it && wrappableInsteadOfChars.find( *(itEnd-1) ) != std::string::npos )
|
|
|
|
--itEnd;
|
2014-02-11 07:32:56 +01:00
|
|
|
}
|
|
|
|
}
|
2017-01-17 18:13:23 +01:00
|
|
|
lines.push_back( indent + std::string( it, itEnd ) + suffix );
|
|
|
|
|
2017-07-02 10:52:29 +02:00
|
|
|
if( indent.size() != attr.indent )
|
|
|
|
indent = std::string( attr.indent, ' ' );
|
2017-01-17 18:13:23 +01:00
|
|
|
it = itNext;
|
2014-02-11 07:32:56 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace Tbc
|
|
|
|
|
|
|
|
#ifdef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE
|
|
|
|
} // end outer namespace
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif // TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED
|