demotool/utilities.hh

275 lines
6.7 KiB
C++
Raw Normal View History

2017-09-30 10:37:45 +02:00
#pragma once
#ifndef REAL_BUILD
# include "externals.hh"
#endif
/*= Utilities ================================================================*/
// Disable next ImGui button(s)
void disableButton( );
// Re-enable ImGui buttons
inline void reenableButtons( )
{
ImGui::PopStyleColor( 3 );
}
/*----------------------------------------------------------------------------*/
2017-10-02 13:51:08 +02:00
#define GL_CHECK( FAIL ) \
2017-10-01 18:51:02 +02:00
do { \
auto err_( glGetError( ) ); \
2017-10-02 13:51:08 +02:00
if ( err_ != GL_NO_ERROR ) FAIL; \
2017-10-01 18:51:02 +02:00
} while ( 0 )
2017-10-02 13:51:08 +02:00
#define GL_ASSERT( ) \
GL_CHECK({ \
fprintf( stderr , "GL error %x in %s:%d\n" , \
err_ , __FILE__ , __LINE__ ); \
abort( ); \
})
2017-10-01 18:51:02 +02:00
/*----------------------------------------------------------------------------*/
2017-09-30 10:37:45 +02:00
// Add some value to an angle, keeping it in [-180;180]
void updateAngle(
__rw__ float& initial ,
__rd__ const float delta
);
// Make a rotation matrix from three YPR angles (in degrees)
void anglesToMatrix(
__rd__ float const* angles ,
__wr__ float* matrix );
/*----------------------------------------------------------------------------*/
// Helpers for finding entries in collections
template< typename T , typename I >
inline auto find(
__rd__ T const& collection ,
__rd__ I const& item )
{
return std::find( collection.begin( ) , collection.end( ) , item );
}
template< typename T , typename I >
inline auto find(
__rw__ T& collection ,
__rd__ I const& item )
{
return std::find( collection.begin( ) , collection.end( ) , item );
}
2017-10-31 09:22:38 +01:00
template< typename T , typename I >
inline bool Contains(
__rd__ T const& collection ,
__rd__ I const& item )
{
return std::find( collection.begin( ) , collection.end( ) , item ) != collection.end( );
}
/*----------------------------------------------------------------------------*/
// Get the absolute path
std::string GetAbsolutePath(
__rd__ std::string const& path );
// Get the absolute parent path for a (possibly relative) path
std::string GetParentPath(
__rd__ std::string const& path );
2017-10-31 09:22:38 +01:00
/*= JSON utilities ===========================================================*/
2017-10-31 14:21:42 +01:00
using T_JSONObject = picojson::value::object;
using T_JSONArray = picojson::value::array;
2017-10-31 09:22:38 +01:00
// Convert a vector to a JSON array
picojson::value jsonVector(
__rd__ float const* vector ,
__rd__ const int nComponents );
// Convert a GLM vec3 to a JSON array
inline picojson::value jsonVector(
__rd__ glm::vec3 const& vector )
{
return jsonVector( &vector.x , 3 );
}
// Add an entry to a JSON object
template< typename T >
inline void jsonAdd(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ T const& v )
{
using T_Entry = std::pair< std::string , picojson::value >;
object.insert( T_Entry( key , picojson::value( v ) ) );
}
extern template void jsonAdd< double >(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ double const& v );
2017-10-31 14:21:42 +01:00
extern template void jsonAdd< T_JSONObject >(
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& v );
extern template void jsonAdd< T_JSONArray >(
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONArray const& v );
2017-10-31 09:22:38 +01:00
extern template void jsonAdd< std::string >(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ std::string const& v );
template< >
void jsonAdd< picojson::value >(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ picojson::value const& v );
template< >
void jsonAdd< int >(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ int const& v );
template< >
void jsonAdd< unsigned >(
2017-10-31 14:21:42 +01:00
__rw__ T_JSONObject& object ,
2017-10-31 09:22:38 +01:00
__rd__ std::string const& key ,
__rd__ unsigned const& v );
/*----------------------------------------------------------------------------*/
// JSON read type error
class X_JsonGetFailed : public std::exception { };
2017-10-31 14:21:42 +01:00
// Get a JSON object field; throws X_JsonGetFailed if the type is incorrect or
// if the field is missing.
template< typename T >
inline T const& jsonGet(
__rd__ T_JSONObject const& object ,
__rd__ char const* const key )
{
const std::string k( key );
if ( object.find( k ) == object.end( ) ) {
throw X_JsonGetFailed( );
}
picojson::value const& v( object.at( k ) );
if ( !v.is< T >( ) ) {
throw X_JsonGetFailed( );
}
return v.get< T >( );
}
2017-10-31 09:22:38 +01:00
// Get a pointer to a JSON object field, or nullptr if the field doesn't exist.
// Throws X_JsonGetFailed if the type is incorrect.
template< typename T >
inline T const* jsonGetOpt(
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* const key )
{
const std::string k( key );
if ( object.find( k ) == object.end( ) ) {
return nullptr;
}
picojson::value const& v( object.at( k ) );
if ( !v.is< T >( ) ) {
throw X_JsonGetFailed( );
}
return &v.get< T >( );
}
// Read a value from a JSON object, if the value actually exists. Fails with
// X_JsonGetFailed if the type is wrong
template< typename T >
inline bool jsonSetFromOpt(
__wr__ T& out ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* const key )
{
T const* ptr( jsonGetOpt< T >( object , key ) );
if ( ptr ) {
out = *ptr;
}
return ptr;
}
// Integer specialization of the above
template< >
inline bool jsonSetFromOpt< int >(
__wr__ int& out ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* const key )
{
double const* ptr( jsonGetOpt< double >( object , key ) );
if ( ptr ) {
out = int( *ptr );
}
return ptr;
}
// Unsigned integer specialization of the above
template< >
inline bool jsonSetFromOpt< unsigned >(
__wr__ unsigned& out ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* const key )
{
double const* ptr( jsonGetOpt< double >( object , key ) );
if ( ptr ) {
out = unsigned( *ptr );
}
return ptr;
}
// 32-bit float specialization of the above
template< >
inline bool jsonSetFromOpt< float >(
__wr__ float& out ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* const key )
{
double const* ptr( jsonGetOpt< double >( object , key ) );
if ( ptr ) {
out = float( *ptr );
}
return ptr;
}
// Read a vector from a JSON array of numbers
void jsonGetVectorOpt(
__wr__ float* vector ,
__rd__ const size_t size ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* key );
// Read a GLM vec3 from a JSON array of numbers
inline void jsonGetVectorOpt(
__wr__ glm::vec3& vector ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* key )
{
jsonGetVectorOpt( &vector.x , 3 , object , key );
}
// Try to load some object using its "bool load( jsonobject )" method.
template< typename T >
inline bool jsonLoadOpt(
__rw__ T& out ,
2017-10-31 14:21:42 +01:00
__rd__ T_JSONObject const& object ,
2017-10-31 09:22:38 +01:00
__rd__ char const* key )
{
try {
2017-10-31 14:21:42 +01:00
auto const* sub( jsonGetOpt< T_JSONObject >( object , key ) );
2017-10-31 09:22:38 +01:00
if ( sub ) {
return out.load( *sub );
}
return true;
} catch ( X_JsonGetFailed const& ) {
return false;
}
}