flags extension

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@23146 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Stefan Csomor
2003-08-23 22:37:11 +00:00
parent 54eba47d04
commit 0886e25499
2 changed files with 104 additions and 34 deletions

View File

@@ -16,36 +16,34 @@
#pragma interface "flags.h" #pragma interface "flags.h"
#endif #endif
#include <bitset> // wxBitset should be applied to an enum, then this can be used like
// wxFlags should be applied to an enum, then this can be used like
// bitwise operators but keeps the type safety and information, the // bitwise operators but keeps the type safety and information, the
// enums must be in a sequence , their value determines the bit position // enums must be in a sequence , their value determines the bit position
// that they represent // that they represent
// The api is made as close as possible to <bitset> // The api is made as close as possible to <bitset>
template <class T> class wxFlags template <class T> class wxBitset
{ {
friend class wxEnumData ; friend class wxEnumData ;
public: public:
// creates a wxFlags<> object with all flags initialized to 0 // creates a wxBitset<> object with all flags initialized to 0
wxFlags() { m_data = 0; } wxBitset() { m_data = 0; }
// created a wxFlags<> object initialized according to the bits of the // created a wxBitset<> object initialized according to the bits of the
// integral value val // integral value val
wxFlags(unsigned long val) { m_data = val ; } wxBitset(unsigned long val) { m_data = val ; }
// copies the content in the new wxFlags<> object from another one // copies the content in the new wxBitset<> object from another one
wxFlags(const wxFlags &src) { m_data = src.m_data; } wxBitset(const wxBitset &src) { m_data = src.m_data; }
// creates a wxFlags<> object that has the specific flag set // creates a wxBitset<> object that has the specific flag set
wxFlags(const T el) { m_data |= 1 << el; } wxBitset(const T el) { m_data |= 1 << el; }
// returns the integral value that the bits of this object represent // returns the integral value that the bits of this object represent
unsigned long to_ulong() const { return m_data ; } unsigned long to_ulong() const { return m_data ; }
// assignment // assignment
wxFlags &operator =(const wxFlags &rhs) wxBitset &operator =(const wxBitset &rhs)
{ {
m_data = rhs.m_data; m_data = rhs.m_data;
return *this; return *this;
@@ -53,7 +51,7 @@ public:
// bitwise or operator, sets all bits that are in rhs and leaves // bitwise or operator, sets all bits that are in rhs and leaves
// the rest unchanged // the rest unchanged
wxFlags &operator |=(const wxFlags &rhs) wxBitset &operator |=(const wxBitset &rhs)
{ {
m_data |= rhs.m_data; m_data |= rhs.m_data;
return *this; return *this;
@@ -61,7 +59,7 @@ public:
// bitwsie exclusive-or operator, toggles the value of all bits // bitwsie exclusive-or operator, toggles the value of all bits
// that are set in bits and leaves all others unchanged // that are set in bits and leaves all others unchanged
wxFlags &operator ^=(const wxFlags &rhs) // difference wxBitset &operator ^=(const wxBitset &rhs) // difference
{ {
m_data ^= rhs.m_data; m_data ^= rhs.m_data;
return *this; return *this;
@@ -69,7 +67,7 @@ public:
// bitwise and operator, resets all bits that are not in rhs and leaves // bitwise and operator, resets all bits that are not in rhs and leaves
// all others unchanged // all others unchanged
wxFlags &operator &=(const wxFlags &rhs) // intersection wxBitset &operator &=(const wxBitset &rhs) // intersection
{ {
m_data &= rhs.m_data; m_data &= rhs.m_data;
return *this; return *this;
@@ -77,47 +75,47 @@ public:
// bitwise or operator, returns a new bitset that has all bits set that set are in // bitwise or operator, returns a new bitset that has all bits set that set are in
// bitset2 or in this bitset // bitset2 or in this bitset
wxFlags operator |(const wxFlags &bitset2) const // union wxBitset operator |(const wxBitset &bitset2) const // union
{ {
wxFlags<T> s; wxBitset<T> s;
s.m_data = m_data | bitset2.m_data; s.m_data = m_data | bitset2.m_data;
return s; return s;
} }
// bitwise exclusive-or operator, returns a new bitset that has all bits set that are set either in // bitwise exclusive-or operator, returns a new bitset that has all bits set that are set either in
// bitset2 or in this bitset but not in both // bitset2 or in this bitset but not in both
wxFlags operator ^(const wxFlags &bitset2) const // difference wxBitset operator ^(const wxBitset &bitset2) const // difference
{ {
wxFlags<T> s; wxBitset<T> s;
s.m_data = m_data ^ bitset2.m_data; s.m_data = m_data ^ bitset2.m_data;
return s; return s;
} }
// bitwise and operator, returns a new bitset that has all bits set that are set both in // bitwise and operator, returns a new bitset that has all bits set that are set both in
// bitset2 and in this bitset // bitset2 and in this bitset
wxFlags operator &(const wxFlags &bitset2) const // intersection wxBitset operator &(const wxBitset &bitset2) const // intersection
{ {
wxFlags<T> s; wxBitset<T> s;
s.m_data = m_data & bitset2.m_data; s.m_data = m_data & bitset2.m_data;
return s; return s;
} }
// sets appropriate the bit to true // sets appropriate the bit to true
wxFlags& set(const T el) //Add element wxBitset& set(const T el) //Add element
{ {
m_data |= 1 << el; m_data |= 1 << el;
return *this; return *this;
} }
// clears the appropriate flag to false // clears the appropriate flag to false
wxFlags& reset(const T el) //remove element wxBitset& reset(const T el) //remove element
{ {
m_data &= ~(1 << el); m_data &= ~(1 << el);
return *this; return *this;
} }
// clear all flags // clear all flags
wxFlags& reset() wxBitset& reset()
{ {
m_data = 0; m_data = 0;
return *this; return *this;
@@ -142,13 +140,13 @@ public:
} }
// true if both have the same flags // true if both have the same flags
bool operator ==(const wxFlags &rhs) const bool operator ==(const wxBitset &rhs) const
{ {
return m_data == rhs.m_data; return m_data == rhs.m_data;
} }
// true if both differ in their flags set // true if both differ in their flags set
bool operator !=(const wxFlags &rhs) const bool operator !=(const wxBitset &rhs) const
{ {
return !operator==(rhs); return !operator==(rhs);
} }
@@ -159,5 +157,11 @@ private :
unsigned long m_data; unsigned long m_data;
}; };
#define WX_DEFINE_FLAGS( flags ) \
struct flags \
{\
flags(long data=0) :m_data(data) {} \
long m_data ;\
} ;
#endif #endif

View File

@@ -143,7 +143,7 @@ private :
// Strawberry, // Strawberry,
// }; // };
// //
// typedef wxFlags<wxFlavor> wxCoupe ; // typedef wxBitset<wxFlavor> wxCoupe ;
// //
// in the implementation file : // in the implementation file :
// //
@@ -164,7 +164,7 @@ private :
void wxSetStringToArray( const wxString &s , wxArrayString &array ) ; void wxSetStringToArray( const wxString &s , wxArrayString &array ) ;
template<typename e> template<typename e>
void wxSetFromString(const wxString &s , wxFlags<e> &data ) void wxSetFromString(const wxString &s , wxBitset<e> &data )
{ {
wxEnumData* edata = wxGetEnumData((e) 0) ; wxEnumData* edata = wxGetEnumData((e) 0) ;
data.reset() ; data.reset() ;
@@ -184,7 +184,7 @@ void wxSetFromString(const wxString &s , wxFlags<e> &data )
} }
template<typename e> template<typename e>
void wxSetToString( wxString &s , const wxFlags<e> &data ) void wxSetToString( wxString &s , const wxBitset<e> &data )
{ {
wxEnumData* edata = wxGetEnumData((e) 0) ; wxEnumData* edata = wxGetEnumData((e) 0) ;
int count = edata->GetEnumCount() ; int count = edata->GetEnumCount() ;
@@ -203,15 +203,15 @@ void wxSetToString( wxString &s , const wxFlags<e> &data )
} }
} }
// if the wxFlags specialization above does not work for all compilers, add this to the WX_IMPLEMENT_SET_STREAMING macro // if the wxBitset specialization above does not work for all compilers, add this to the WX_IMPLEMENT_SET_STREAMING macro
// template<> const wxTypeInfo* wxGetTypeInfo( SetName * ){ static wxEnumTypeInfo s_typeInfo(wxT_SET , &s_enumData##e) ; return &s_typeInfo ; } // template<> const wxTypeInfo* wxGetTypeInfo( SetName * ){ static wxEnumTypeInfo s_typeInfo(wxT_SET , &s_enumData##e) ; return &s_typeInfo ; }
#define WX_IMPLEMENT_SET_STREAMING(SetName,e) \ #define WX_IMPLEMENT_SET_STREAMING(SetName,e) \
template<> void wxStringReadValue(const wxString &s , wxFlags<e> &data ) \ template<> void wxStringReadValue(const wxString &s , wxBitset<e> &data ) \
{ \ { \
wxSetFromString( s , data ) ; \ wxSetFromString( s , data ) ; \
} \ } \
template<> void wxStringWriteValue( wxString &s , const wxFlags<e> &data ) \ template<> void wxStringWriteValue( wxString &s , const wxBitset<e> &data ) \
{ \ { \
wxSetToString( s , data ) ; \ wxSetToString( s , data ) ; \
} \ } \
@@ -222,6 +222,72 @@ template<> const wxTypeInfo* wxGetTypeInfo( SetName * ) \
static wxEnumTypeInfo s_typeInfo(wxT_SET , &s_enumData##e , &wxToStringConverter<SetName> , &wxFromStringConverter<SetName> , &ToLong##SetName , &FromLong##SetName, #SetName ) ; return &s_typeInfo ; \ static wxEnumTypeInfo s_typeInfo(wxT_SET , &s_enumData##e , &wxToStringConverter<SetName> , &wxFromStringConverter<SetName> , &ToLong##SetName , &FromLong##SetName, #SetName ) ; return &s_typeInfo ; \
} }
template<typename e>
void wxFlagsFromString(const wxString &s , e &data )
{
wxEnumData* edata = wxGetEnumData((e*) 0) ;
data.m_data = 0 ;
wxArrayString array ;
wxSetStringToArray( s , array ) ;
wxString flag;
for ( int i = 0 ; i < array.Count() ; ++i )
{
flag = array[i] ;
int ivalue ;
if ( edata->HasEnumMemberValue( flag , &ivalue ) )
{
data.m_data |= ivalue ;
}
}
}
template<typename e>
void wxFlagsToString( wxString &s , const e& data )
{
wxEnumData* edata = wxGetEnumData((e*) 0) ;
int count = edata->GetEnumCount() ;
int i ;
s.Clear() ;
long dataValue = data.m_data ;
for ( i = 0 ; i < count ; i++ )
{
int value = edata->GetEnumMemberValueByIndex(i) ;
// make this to allow for multi-bit constants to work
if ( value && ( dataValue & value ) == value )
{
// clear the flags we just set
dataValue &= ~value ;
// this could also be done by the templated calls
if ( !s.IsEmpty() )
s +="|" ;
s += edata->GetEnumMemberNameByIndex(i) ;
}
}
}
#define WX_BEGIN_FLAGS( e ) \
wxEnumMemberData s_enumDataMembers##e[] = {
#define WX_FLAGS_MEMBER( v ) { #v, v } ,
#define WX_END_FLAGS( e ) { NULL , 0 } } ; \
wxEnumData s_enumData##e( s_enumDataMembers##e ) ; \
wxEnumData *wxGetEnumData(e*) { return &s_enumData##e ; } \
template<> void wxStringReadValue(const wxString &s , e &data ) \
{ \
wxFlagsFromString<e>( s , data ) ; \
} \
template<> void wxStringWriteValue( wxString &s , const e& data ) \
{ \
wxFlagsToString<e>( s , data ) ; \
} \
void FromLong##e( long data , wxxVariant& result ) { result = wxxVariant(e(data)) ;} \
void ToLong##e( const wxxVariant& data , long &result ) { result = (long) data.Get<e>().m_data ;} \
template<> const wxTypeInfo* wxGetTypeInfo( e * ) \
{ \
static wxEnumTypeInfo s_typeInfo(wxT_SET , &s_enumData##e , &wxToStringConverter<e> , &wxFromStringConverter<e> , &ToLong##e , &FromLong##e, #e ) ; return &s_typeInfo ; \
}
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Type Information // Type Information
@@ -247,7 +313,7 @@ enum wxTypeKind
wxT_FLOAT, wxT_FLOAT,
wxT_DOUBLE, wxT_DOUBLE,
wxT_STRING, // must be wxString wxT_STRING, // must be wxString
wxT_SET, // must be wxFlags<> template wxT_SET, // must be wxBitset<> template
wxT_ENUM, wxT_ENUM,
wxT_CUSTOM, // user defined type (e.g. wxPoint) wxT_CUSTOM, // user defined type (e.g. wxPoint)