2024-07-28 22:42:16 +08:00
# pragma once
# include "YYCCInternal.hpp"
# include "Constraints.hpp"
# include "EncodingHelper.hpp"
2024-07-29 16:58:52 +08:00
# include "ParserHelper.hpp"
2024-07-28 22:42:16 +08:00
# include <functional>
# include <vector>
# include <map>
# include <stdexcept>
namespace YYCC : : ArgParser {
class ArgumentList {
public :
static ArgumentList CreateFromStd ( int argc , char * argv [ ] ) ;
# if YYCC_OS == YYCC_OS_WINDOWS
static ArgumentList CreateFromWin32 ( ) ;
# endif
private :
ArgumentList ( std : : vector < yycc_u8string > & & arguments ) ;
2024-07-29 16:58:52 +08:00
public :
ArgumentList ( const ArgumentList & ) = default ;
ArgumentList & operator = ( const ArgumentList & ) = default ;
ArgumentList ( ArgumentList & & ) = default ;
ArgumentList & operator = ( ArgumentList & & ) = default ;
2024-07-28 22:42:16 +08:00
public :
void Prev ( ) ;
void Next ( ) ;
2024-07-29 16:58:52 +08:00
const yycc_u8string & Current ( ) const ;
2024-07-28 22:42:16 +08:00
bool IsSwitch (
bool * is_long_name = nullptr ,
2024-07-29 16:58:52 +08:00
yycc_u8string * long_name = nullptr ,
2024-07-28 22:42:16 +08:00
yycc_char8_t * short_name = nullptr ) const ;
2024-07-29 16:58:52 +08:00
bool IsValue ( yycc_u8string * val = nullptr ) const ;
2024-07-28 22:42:16 +08:00
bool IsEOF ( ) const ;
void Reset ( ) ;
2024-07-29 16:58:52 +08:00
private :
bool IsLongNameSwitch ( yycc_u8string * name_part = nullptr ) const ;
bool IsShortNameSwitch ( yycc_char8_t * name_part = nullptr ) const ;
2024-07-28 22:42:16 +08:00
private :
std : : vector < yycc_u8string > m_Arguments ;
std : : vector < yycc_u8string > : : const_iterator m_ArgumentsIterator ;
} ;
class AbstractArgument {
friend class OptionContext ;
2024-07-29 16:58:52 +08:00
// Long name and short name constants and checker.
2024-07-28 22:42:16 +08:00
public :
static const yycc_u8string DOUBLE_DASH ;
static const yycc_char8_t DASH ;
static const yycc_char8_t NO_SHORT_NAME ;
static const yycc_char8_t MIN_SHORT_NAME ;
static const yycc_char8_t MAX_SHORT_NAME ;
2024-07-29 16:58:52 +08:00
static bool IsLegalShortName ( yycc_char8_t short_name ) ;
static bool IsLegalLongName ( const yycc_u8string_view & long_name ) ;
// Constructor & destructor
2024-07-28 22:42:16 +08:00
public :
AbstractArgument (
const yycc_char8_t * long_name , yycc_char8_t short_name = AbstractArgument : : NO_SHORT_NAME ,
const yycc_char8_t * description = nullptr , const yycc_char8_t * argument_example = nullptr ,
bool is_optional = false ) ;
virtual ~ AbstractArgument ( ) ;
protected :
virtual bool Parse ( ArgumentList & al ) = 0 ;
virtual void Reset ( ) = 0 ;
public :
bool HasLongName ( ) const ;
const yycc_u8string & GetLongName ( ) const ;
bool HasShortName ( ) const ;
yycc_char8_t GetShortName ( ) const ;
bool HasDescription ( ) const ;
const yycc_u8string & GetDescription ( ) const ;
bool HasArgumentExample ( ) const ;
const yycc_u8string & GetArgumentExample ( ) const ;
bool IsOptional ( ) const ;
private :
yycc_u8string m_LongName ;
yycc_char8_t m_ShortName ;
yycc_u8string m_Description ;
yycc_u8string m_ArgumentExample ;
bool m_IsOptional ;
public :
bool IsCaptured ( ) const ;
private :
void SetCaptured ( bool is_captured ) ;
bool m_IsCaptured ;
} ;
class OptionContext {
public :
OptionContext (
const yycc_char8_t * summary , const yycc_char8_t * description ,
std : : initializer_list < AbstractArgument * > arguments ) ;
~ OptionContext ( ) ;
public :
2024-07-29 16:58:52 +08:00
bool Parse ( ArgumentList & al ) ;
2024-07-28 22:42:16 +08:00
void Reset ( ) ;
private :
yycc_u8string m_Summary ;
yycc_u8string m_Description ;
std : : vector < AbstractArgument * > m_Arguments ;
std : : map < yycc_u8string , AbstractArgument * > m_LongNameMap ;
std : : map < yycc_char8_t , AbstractArgument * > m_ShortNameMap ;
} ;
# pragma region Argument Presets
template < typename _Ty , std : : enable_if_t < std : : is_arithmetic_v < _Ty > & & ! std : : is_same_v < _Ty , bool > , int > = 0 >
class NumberArgument : public AbstractArgument {
public :
NumberArgument (
const yycc_char8_t * long_name , yycc_char8_t short_name ,
const yycc_char8_t * description = nullptr , const yycc_char8_t * argument_example = nullptr ,
bool is_optional = false ,
Constraints : : Constraint < _Ty > constraint = Constraints : : Constraint < _Ty > { } ) :
AbstractArgument ( long_name , short_name , description , argument_example , is_optional ) , m_Data ( ) , m_Constraint ( constraint ) { }
virtual ~ NumberArgument ( ) { }
public :
_Ty Get ( ) const {
if ( ! IsCaptured ( ) ) throw std : : runtime_error ( " try fetching data from a not captured argument. " ) ;
return m_Data ;
}
protected :
2024-07-29 16:58:52 +08:00
virtual bool Parse ( ArgumentList & al ) override {
// try get corresponding value
yycc_u8string strval ;
al . Next ( ) ;
if ( al . IsEOF ( ) | | ! al . IsValue ( & strval ) ) {
al . Prev ( ) ;
return false ;
}
// try parsing value
if ( ! YYCC : : ParserHelper : : TryParse < _Ty > ( strval , m_Data ) ) return false ;
// check constraint
if ( m_Constraint . IsValid ( ) & & ! m_Constraint . m_CheckFct ( m_Data ) )
return false ;
// okey
return true ;
}
virtual void Reset ( ) override {
std : : memset ( & m_Data , 0 , sizeof ( m_Data ) ) ;
}
2024-07-28 22:42:16 +08:00
protected :
_Ty m_Data ;
Constraints : : Constraint < _Ty > m_Constraint ;
} ;
class SwitchArgument : public AbstractArgument {
public :
SwitchArgument (
const yycc_char8_t * long_name , yycc_char8_t short_name ,
const yycc_char8_t * description = nullptr , const yycc_char8_t * argument_example = nullptr ) :
AbstractArgument ( long_name , short_name , description , argument_example , true ) , m_Data ( false ) { } // bool switch must be optional, because it is false if no given switch.
virtual ~ SwitchArgument ( ) { }
public :
bool Get ( ) const { return m_Data ; }
protected :
2024-07-29 16:58:52 +08:00
virtual bool Parse ( ArgumentList & al ) override {
m_Data = true ;
return true ;
}
2024-07-28 22:42:16 +08:00
virtual void Reset ( ) override { m_Data = false ; }
protected :
bool m_Data ;
} ;
class StringArgument : public AbstractArgument {
public :
StringArgument (
const yycc_char8_t * long_name , yycc_char8_t short_name ,
const yycc_char8_t * description = nullptr , const yycc_char8_t * argument_example = nullptr ,
bool is_optional = false ,
2024-07-29 16:58:52 +08:00
Constraints : : Constraint < yycc_u8string > constraint = Constraints : : Constraint < yycc_u8string > { } ) :
2024-07-28 22:42:16 +08:00
AbstractArgument ( long_name , short_name , description , argument_example , is_optional ) , m_Data ( ) , m_Constraint ( constraint ) { }
virtual ~ StringArgument ( ) { }
public :
const yycc_u8string & Get ( ) const {
if ( ! IsCaptured ( ) ) throw std : : runtime_error ( " try fetching data from a not captured argument. " ) ;
return m_Data ;
}
protected :
2024-07-29 16:58:52 +08:00
virtual bool Parse ( ArgumentList & al ) override {
// try get corresponding value
al . Next ( ) ;
if ( al . IsEOF ( ) | | ! al . IsValue ( & m_Data ) ) {
al . Prev ( ) ;
return false ;
}
// check constraint
if ( m_Constraint . IsValid ( ) & & ! m_Constraint . m_CheckFct ( m_Data ) )
return false ;
// okey
return true ;
}
virtual void Reset ( ) override {
m_Data . clear ( ) ;
}
2024-07-28 22:42:16 +08:00
protected :
yycc_u8string m_Data ;
2024-07-29 16:58:52 +08:00
Constraints : : Constraint < yycc_u8string > m_Constraint ;
2024-07-28 22:42:16 +08:00
} ;
# pragma endregion
}