demotool/syncoverrides.cc

398 lines
9.9 KiB
C++

#include "externals.hh"
#include "globals.hh"
#include "syncoverrides.hh"
#include <ebcl/SRDParser.hh>
using namespace sov;
#define M_SETOPT_( FIELD , VAR ) \
if ( FIELD ) { return false; } \
FIELD = (VAR); \
return true
/*= PARSER DEFINITIONS =========================================================*/
namespace {
using namespace ebcl;
using SP_Section = T_SharedPtr< T_SyncOverrideSection >;
bool EnterSection_( T_SRDParserData const& data )
{
*( data.targetData ) = NewShared< T_SyncOverrideSection >( (*data.input)[ 1 ].stringValue( ) );
return true;
}
bool ExitSection_( T_SRDParserData const& data )
{
auto& section( data.currentData->value< SP_Section >( ) );
auto& parent( data.targetData->value< SP_Section >( ) );
if ( section->overrides.empty( ) && section->subsections.empty( ) ) {
T_StringBuilder sb;
sb << "empty section '" << section->title << "'";
data.errors.add( std::move( sb ) , (*data.input)[ 0 ] );
} else {
parent->subsections.add( section.makeOwned( ) );
}
return true;
}
/*------------------------------------------------------------------------------*/
using SP_Float = T_SharedPtr< A_Float >;
bool EnterFloat1_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
SP_Float ptr{ NewShared< T_Float >(
input[ 1 ].stringValue( ) , input[ 2 ].stringValue( ) ) };
*( data.targetData ) = std::move( ptr );
return true;
}
bool EnterFloat2_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
SP_Float ptr{ NewShared< T_Float2 >(
input[ 1 ].stringValue( ) , input[ 2 ].stringValue( ) ,
input[ 3 ].stringValue( ) ) };
*( data.targetData ) = std::move( ptr );
return true;
}
bool EnterFloat3_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
SP_Float ptr{ NewShared< T_Float3 >(
input[ 1 ].stringValue( ) , input[ 2 ].stringValue( ) ,
input[ 3 ].stringValue( ) , input[ 4 ].stringValue( ) ) };
*( data.targetData ) = std::move( ptr );
return true;
}
bool EnterFloat4_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
SP_Float ptr{ NewShared< T_Float4 >(
input[ 1 ].stringValue( ) , input[ 2 ].stringValue( ) ,
input[ 3 ].stringValue( ) , input[ 4 ].stringValue( ) ,
input[ 5 ].stringValue( ) ) };
*( data.targetData ) = std::move( ptr );
return true;
}
bool AddFloat_( T_SRDParserData const& data )
{
auto& fl( data.currentData->value< SP_Float >( ) );
auto& parent( data.targetData->value< SP_Section >( ) );
if ( fl->min( ) > fl->max( ) ) {
data.errors.add( "invalid bounds" , (*data.input)[ 0 ] );
}
parent->overrides.add( fl.makeOwned( ) );
return true;
}
/*------------------------------------------------------------------------------*/
bool FloatSetSlider_( T_SRDParserData const& data )
{
data.currentData->value< SP_Float >( )->setSlider( );
return true;
}
bool FloatSetMin_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
if ( !data.currentData->value< SP_Float >( )->setMin( input[ 1 ].floatValue( ) ) ) {
data.errors.add( "duplicate minimal value" , (*data.input)[ 0 ] );
}
return true;
}
bool FloatSetMax_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
if ( !data.currentData->value< SP_Float >( )->setMax( input[ 1 ].floatValue( ) ) ) {
data.errors.add( "duplicate maximal value" , (*data.input)[ 0 ] );
}
return true;
}
bool FloatSetStep_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
const auto v( input[ 1 ].floatValue( ) );
if ( v <= 0 ) {
data.errors.add( "invalid step value" , (*data.input)[ 1 ] );
} else if ( !data.currentData->value< SP_Float >( )->setStep( v ) ) {
data.errors.add( "duplicate step value" , (*data.input)[ 0 ] );
}
return true;
}
bool FloatSetPower_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
const auto v( input[ 1 ].floatValue( ) );
if ( v <= 0 ) {
data.errors.add( "invalid power value" , (*data.input)[ 1 ] );
} else if ( !data.currentData->value< SP_Float >( )->setPower( v ) ) {
data.errors.add( "duplicate power value" , (*data.input)[ 0 ] );
}
return true;
}
bool FloatSetDecimals_( T_SRDParserData const& data )
{
auto const& input( *( data.input ) );
const auto v( input[ 1 ].longValue( ) );
if ( v < 0 || v > 10 ) {
data.errors.add( "invalid decimals value" , (*data.input)[ 1 ] );
} else if ( !data.currentData->value< SP_Float >( )->setDecimals( v ) ) {
data.errors.add( "duplicate decimals value" , (*data.input)[ 0 ] );
}
return true;
}
} // namespace
/*------------------------------------------------------------------------------*/
ebcl::T_SRDParserConfig sov::GetParserConfig( )
{
using namespace ebcl;
using namespace ebcl::SRD;
T_SRDParserDefs defs( "root" );
defs << OnStart( []( T_SRDParserData const& data ) -> bool {
*( data.currentData ) = NewShared< T_SyncOverrideSection >( "*" );
return true;
} );
defs.context( "root" )
<< ( Rule() << "section" << Text( ) << EnterContext( "section" )
<< OnEnter( EnterSection_ )
<< OnExit( ExitSection_ ) )
;
defs.context( "section" )
<< ( Rule() << "section" << Text( ) << EnterContext( "section" ) )
// Floating point controls
<< ( Rule() << "float" << Text( ) << Word( )
<< EnterContext( "float" )
<< OnEnter( EnterFloat1_ )
<< OnExit( AddFloat_ ) )
<< ( Rule() << "float2" << Text( )
<< ( SRD::Times( 2 ) << Word( ) )
<< EnterContext( "float" )
<< OnEnter( EnterFloat2_ )
<< OnExit( AddFloat_ ) )
<< ( Rule() << "float3" << Text( )
<< ( SRD::Times( 3 ) << Word( ) )
<< EnterContext( "float" )
<< OnEnter( EnterFloat3_ )
<< OnExit( AddFloat_ ) )
<< ( Rule() << "float4" << Text( )
<< ( SRD::Times( 4 ) << Word( ) )
<< EnterContext( "float" )
<< OnEnter( EnterFloat4_ )
<< OnExit( AddFloat_ ) )
;
// Floating point control parameters
defs.context( "float" )
<< ( Rule() << "slider" << FloatSetSlider_ )
<< ( Rule() << "min" << Numeric() << FloatSetMin_ )
<< ( Rule() << "max" << Numeric() << FloatSetMax_ )
<< ( Rule() << "step" << Numeric() << FloatSetStep_ )
<< ( Rule() << "power" << Numeric() << FloatSetPower_ )
<< ( Rule() << "decimals" << Integer() << FloatSetDecimals_ )
;
return T_SRDParserConfig{ defs };
}
/*= A_Float ====================================================================*/
bool A_Float::setMin(
const float v ) noexcept
{
M_SETOPT_( min_ , v );
}
bool A_Float::setMax(
const float v ) noexcept
{
M_SETOPT_( max_ , v );
}
bool A_Float::setStep(
const float v ) noexcept
{
assert( v > 0 );
M_SETOPT_( step_ , v );
}
bool A_Float::setDecimals(
const uint32_t n ) noexcept
{
assert( n <= 100 );
if ( decimals_ ) {
return false;
}
T_StringBuilder sb;
sb << "%." << n << 'f' << '\0';
assert( sb.size( ) < 12 );
decimals_.setNew( );
for ( auto i = 0u ; i < sb.size( ) ; i ++ ) {
decimals_->add( sb.data( )[ i ] );
}
return true;
}
bool A_Float::setPower(
const float v ) noexcept
{
assert( v > 0 );
M_SETOPT_( power_ , v );
}
void A_Float::setSlider( ) noexcept
{
slider_ = true;
}
/*= T_Float ====================================================================*/
T_Float::T_Float(
T_String const& input ,
T_String const& title ) noexcept
: A_Float( "float" , title )
{
inputs_.add( input );
}
void T_Float::makeEditWidgets( ) noexcept
{
using namespace ImGui;
float v[ 1 ] = {
Globals::Sync( ).inputs( )[ inputPos_[ 0 ] ]
};
const bool changed( slider( )
? SliderFloat( "" , v , min( ) , max( ) , decimals( ) , power( ) )
: DragFloat( "" , v , step( ) , min( ) , max( ) , decimals( ) , power( ) ) );
if ( changed ) {
Globals::Sync( ).inputs( )[ inputPos_[ 0 ] ] = v[ 0 ];
}
}
/*= T_Float2 ===================================================================*/
T_Float2::T_Float2(
T_String const& input0 ,
T_String const& input1 ,
T_String const& title ) noexcept
: A_Float( "float" , title )
{
inputs_.add( input0 );
inputs_.add( input1 );
}
void T_Float2::makeEditWidgets( ) noexcept
{
using namespace ImGui;
auto& sinp( Globals::Sync( ).inputs( ) );
float v[ 2 ];
for ( auto i = 0 ; i < 2 ; i ++ ) {
v[ i ] = sinp[ inputPos_[ i ] ];
}
const bool changed( slider( )
? SliderFloat2( "" , v , min( ) , max( ) , decimals( ) , power( ) )
: DragFloat2( "" , v , step( ) , min( ) , max( ) , decimals( ) , power( ) ) );
if ( changed ) {
for ( auto i = 0 ; i < 2 ; i ++ ) {
sinp[ inputPos_[ i ] ] = v[ i ];
}
}
}
/*= T_Float3 ===================================================================*/
T_Float3::T_Float3(
T_String const& input0 ,
T_String const& input1 ,
T_String const& input2 ,
T_String const& title ) noexcept
: A_Float( "float" , title )
{
inputs_.add( input0 );
inputs_.add( input1 );
inputs_.add( input2 );
}
void T_Float3::makeEditWidgets( ) noexcept
{
using namespace ImGui;
auto& sinp( Globals::Sync( ).inputs( ) );
float v[ 3 ];
for ( auto i = 0 ; i < 3 ; i ++ ) {
v[ i ] = sinp[ inputPos_[ i ] ];
}
const bool changed( slider( )
? SliderFloat3( "" , v , min( ) , max( ) , decimals( ) , power( ) )
: DragFloat3( "" , v , step( ) , min( ) , max( ) , decimals( ) , power( ) ) );
if ( changed ) {
for ( auto i = 0 ; i < 3 ; i ++ ) {
sinp[ inputPos_[ i ] ] = v[ i ];
}
}
}
/*= T_Float4 ===================================================================*/
T_Float4::T_Float4(
T_String const& input0 ,
T_String const& input1 ,
T_String const& input2 ,
T_String const& input3 ,
T_String const& title ) noexcept
: A_Float( "float" , title )
{
inputs_.add( input0 );
inputs_.add( input1 );
inputs_.add( input2 );
inputs_.add( input3 );
}
void T_Float4::makeEditWidgets( ) noexcept
{
using namespace ImGui;
auto& sinp( Globals::Sync( ).inputs( ) );
float v[ 4 ];
for ( auto i = 0 ; i < 4 ; i ++ ) {
v[ i ] = sinp[ inputPos_[ i ] ];
}
const bool changed( slider( )
? SliderFloat4( "" , v , min( ) , max( ) , decimals( ) , power( ) )
: DragFloat4( "" , v , step( ) , min( ) , max( ) , decimals( ) , power( ) ) );
if ( changed ) {
for ( auto i = 0 ; i < 4 ; i ++ ) {
sinp[ inputPos_[ i ] ] = v[ i ];
}
}
}