1
0
forked from Alepha/Alepha
Files
Alepha/IOStreams/StreamState.h

131 lines
2.5 KiB
C++

static_assert( __cplusplus > 2020'99 );
#pragma once
#include <ios>
#include <exception>
#include <functional>
#include <boost/noncopyable.hpp>
#include <Alepha/Alepha.h>
namespace Alepha::Hydrogen::IOStreams ::detail:: StreamState_m
{
inline namespace exports
{
template< typename >
class StreamState;
}
template< typename Type >
class exports::StreamState : boost::noncopyable
{
private:
const int index= std::ios::xalloc();
std::function< Type () > build;
public:
explicit
StreamState( const std::function< Type () > build )
: build( build ) {}
private:
static Type *&
get_ptr( std::ios_base &ios, const int idx )
{
return reinterpret_cast< Type *& >( ios.pword( idx ) );
}
Type *&
get_ptr( std::ios_base &ios )
{
return get_ptr( ios, index );
}
static void
destroy( std::ios_base &ios, const int idx )
{
delete get_ptr( ios, idx );
get_ptr( ios, idx )= nullptr;
}
static void
callback_impl( const std::ios_base::event event, std::ios_base &ios, const int idx )
{
if( event == std::ios_base::erase_event ) destroy( ios, idx );
else if( event == std::ios_base::imbue_event )
{
// Nothing to do... until I develop locale support.
}
else if( event == std::ios_base::copyfmt_event )
{
get_ptr( ios, idx )= new Type{ *get_ptr( ios, idx ) };
}
}
static void
callback( const std::ios_base::event event, std::ios_base &ios, const int idx ) noexcept
{
return callback_impl( event, ios, idx );
}
void
init( std::ios_base &ios )
{
if( not ios.iword( index ) )
{
ios.iword( index )= 1;
ios.register_callback( callback, index );
}
auto *&ptr= get_ptr( ios, index );
if( not ptr ) ptr= new Type{ build() };
}
public:
Type &
get( std::ios_base &ios )
{
init( ios );
return *get_ptr( ios );
}
void
setDefault( const Type value )
{
build= [value] { return value; };
}
struct Setter
{
StreamState *state;
const Type val;
friend std::ostream &
operator << ( std::ostream &os, const Setter &s )
{
s.state->get( os )= s.val;
return os;
}
friend std::istream &
operator >> ( std::istream &is, const Setter &s )
{
s.state->get( is )= s.val;
return is;
}
};
auto
makeSetter( const Type val )
{
return Setter{ this, val };
}
};
}
namespace Alepha::Hydrogen::IOStreams::inline exports::inline StreamState_m
{
using namespace detail::StreamState_m::exports;
}