forked from Alepha/Alepha
Word wrapping as stream operations now.
This commit is contained in:
145
word_wrap.cpp
145
word_wrap.cpp
@ -8,6 +8,7 @@ static_assert( __cplusplus > 2020 );
|
|||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <iterator>
|
#include <iterator>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#include "evaluation_helpers.h"
|
#include "evaluation_helpers.h"
|
||||||
|
|
||||||
@ -36,6 +37,41 @@ namespace Alepha::Cavorite ::detail:: word_wrap
|
|||||||
result << std::move( word );
|
result << std::move( word );
|
||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct WordWrapStreambuf
|
||||||
|
: public std::streambuf
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
std::unique_ptr< std::streambuf > ownership;
|
||||||
|
std::streambuf *underlying= nullptr;
|
||||||
|
|
||||||
|
std::size_t maximumWidth= 0;
|
||||||
|
std::size_t nextLineOffset= 0;
|
||||||
|
std::size_t currentLineLength= 0;
|
||||||
|
|
||||||
|
std::string currentWord;
|
||||||
|
|
||||||
|
void writeChar( const char ch );
|
||||||
|
|
||||||
|
void drain();
|
||||||
|
|
||||||
|
public:
|
||||||
|
int
|
||||||
|
overflow( const int ch ) override
|
||||||
|
{
|
||||||
|
if( ch == EOF ) throw std::logic_error( "EOF!" );
|
||||||
|
writeChar( ch );
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::streamsize
|
||||||
|
xsputn( const char *const data, const std::streamsize amt ) override
|
||||||
|
{
|
||||||
|
for( std::streamsize i= 0; i< amt; ++i ) overflow( data[ i ] );
|
||||||
|
return amt;
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -80,62 +116,69 @@ namespace Alepha::Cavorite ::detail:: word_wrap
|
|||||||
std::string
|
std::string
|
||||||
exports::wordWrap( const std::string &text, const std::size_t width, const std::size_t nextLineOffset )
|
exports::wordWrap( const std::string &text, const std::size_t width, const std::size_t nextLineOffset )
|
||||||
{
|
{
|
||||||
#if NEVER
|
|
||||||
auto putWord= [width, nextLineOffset]( std::string &&word, std::string &line, const std::size_t lineLength )
|
|
||||||
{
|
|
||||||
std::ostringstream out;
|
|
||||||
const auto rv= applyWordToLine( width, nextLineOffset, lineLength, std::move( word ), out );
|
|
||||||
line+= std::move( out ).str();
|
|
||||||
return rv;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::string result;
|
|
||||||
std::string word;
|
|
||||||
std::size_t lineLength= 0;
|
|
||||||
for( const char ch: text )
|
|
||||||
{
|
|
||||||
if( ch == '\n' )
|
|
||||||
{
|
|
||||||
const auto prev= lineLength;
|
|
||||||
const auto size= word.size();
|
|
||||||
lineLength= putWord( std::move( word ), result, lineLength );
|
|
||||||
word.clear();
|
|
||||||
result+= '\n';
|
|
||||||
if( lineLength == prev + size )
|
|
||||||
{
|
|
||||||
std::fill_n( back_inserter( result ), nextLineOffset, ' ' );
|
|
||||||
lineLength= nextLineOffset;
|
|
||||||
}
|
|
||||||
else lineLength= 0;
|
|
||||||
}
|
|
||||||
else if( ch == ' ' )
|
|
||||||
{
|
|
||||||
lineLength= putWord( std::move( word ), result, lineLength );
|
|
||||||
word.clear();
|
|
||||||
if( lineLength < width )
|
|
||||||
{
|
|
||||||
result+= ' ';
|
|
||||||
lineLength++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else word+= ch;
|
|
||||||
}
|
|
||||||
if( not word.empty() ) std::ignore= putWord( std::move( word ), result, lineLength );
|
|
||||||
return result;
|
|
||||||
#else
|
|
||||||
std::ostringstream oss;
|
std::ostringstream oss;
|
||||||
|
|
||||||
WordWrapStreambuf buf;
|
oss << StartWrap{ width, nextLineOffset };
|
||||||
buf.maximumWidth= width;
|
|
||||||
buf.nextLineOffset= nextLineOffset;
|
|
||||||
buf.underlying= static_cast< std::ostream & >( oss ).rdbuf();
|
|
||||||
static_cast< std::ostream & >( oss ).rdbuf( &buf );
|
|
||||||
|
|
||||||
oss << text;
|
oss << text;
|
||||||
buf.drain();
|
oss << EndWrap;
|
||||||
|
|
||||||
auto rv= std::move( oss ).str();
|
auto rv= std::move( oss ).str();
|
||||||
return rv;
|
return rv;
|
||||||
#endif
|
}
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
const auto wrapperIndex= std::ios_base::xalloc();
|
||||||
|
|
||||||
|
void
|
||||||
|
releaseWrapper( std::ios_base &ios )
|
||||||
|
{
|
||||||
|
auto *const streambuf= static_cast< WordWrapStreambuf * >( ios.pword( wrapperIndex ) );
|
||||||
|
streambuf->drain();
|
||||||
|
delete streambuf;
|
||||||
|
ios.pword( wrapperIndex )= nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
wordwrapCallback( const std::ios_base::event event, std::ios_base &ios, const int idx ) noexcept
|
||||||
|
{
|
||||||
|
#pragma GCC diagnostic push
|
||||||
|
#pragma GCC diagnostic ignored "-Wterminate"
|
||||||
|
|
||||||
|
if( wrapperIndex != idx ) throw std::logic_error( "Must only work with the word wrap index." );
|
||||||
|
|
||||||
|
if( not ios.pword( wrapperIndex ) ) return;
|
||||||
|
|
||||||
|
if( event == std::ios_base::erase_event ) releaseWrapper( ios );
|
||||||
|
else if( event == std::ios_base::imbue_event ) {}
|
||||||
|
else if( event == std::ios_base::copyfmt_event ) throw std::runtime_error{ "Can't copy?" };
|
||||||
|
|
||||||
|
#pragma GCC diagnostic pop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream &
|
||||||
|
impl::operator << ( std::ostream &os, EndWrap_t )
|
||||||
|
{
|
||||||
|
releaseWrapper( os );
|
||||||
|
return os;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostream &
|
||||||
|
impl::operator << ( std::ostream &os, StartWrap args )
|
||||||
|
{
|
||||||
|
auto buf= std::make_unique< WordWrapStreambuf >();
|
||||||
|
buf->maximumWidth= args.width;
|
||||||
|
buf->nextLineOffset= args.nextLineOffset;
|
||||||
|
buf->underlying= os.rdbuf( buf.get() );
|
||||||
|
auto &state= os.iword( wrapperIndex );
|
||||||
|
if( not state )
|
||||||
|
{
|
||||||
|
state= 1;
|
||||||
|
os.register_callback( wordwrapCallback, wrapperIndex );
|
||||||
|
}
|
||||||
|
os.pword( wrapperIndex )= buf.release();
|
||||||
|
|
||||||
|
return os;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
46
word_wrap.h
46
word_wrap.h
@ -13,42 +13,22 @@ namespace Alepha::inline Cavorite ::detail:: word_wrap
|
|||||||
{
|
{
|
||||||
std::string wordWrap( const std::string &text, std::size_t width, std::size_t nextLineOffset= 0 );
|
std::string wordWrap( const std::string &text, std::size_t width, std::size_t nextLineOffset= 0 );
|
||||||
|
|
||||||
class WordWrapStreambuf;
|
struct StartWrap
|
||||||
|
{
|
||||||
|
std::size_t width;
|
||||||
|
std::size_t nextLineOffset;
|
||||||
|
|
||||||
|
explicit StartWrap( const std::size_t width, const std::size_t nextLineOffset= 0 ) : width( width ), nextLineOffset( nextLineOffset ) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr struct EndWrap_t {} EndWrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
class exports::WordWrapStreambuf
|
inline namespace impl
|
||||||
: public std::streambuf
|
|
||||||
{
|
{
|
||||||
public:
|
std::ostream &operator << ( std::ostream &, StartWrap );
|
||||||
std::streambuf *underlying= nullptr;
|
std::ostream &operator << ( std::ostream &, EndWrap_t );
|
||||||
|
}
|
||||||
std::size_t maximumWidth= 0;
|
|
||||||
std::size_t nextLineOffset= 0;
|
|
||||||
std::size_t currentLineLength= 0;
|
|
||||||
|
|
||||||
std::string currentWord;
|
|
||||||
|
|
||||||
void writeChar( const char ch );
|
|
||||||
|
|
||||||
void drain();
|
|
||||||
|
|
||||||
public:
|
|
||||||
int
|
|
||||||
overflow( const int ch ) override
|
|
||||||
{
|
|
||||||
if( ch == EOF ) throw std::logic_error( "EOF!" );
|
|
||||||
writeChar( ch );
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::streamsize
|
|
||||||
xsputn( const char *const data, const std::streamsize amt ) override
|
|
||||||
{
|
|
||||||
for( std::streamsize i= 0; i< amt; ++i ) overflow( data[ i ] );
|
|
||||||
return amt;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace Alepha::Cavorite::inline exports::inline word_wrap
|
namespace Alepha::Cavorite::inline exports::inline word_wrap
|
||||||
|
Reference in New Issue
Block a user