1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/any.cpp
3 // Purpose: wxAny class, container for any type
4 // Author: Jaakko Salli
8 // Copyright: (c) wxWidgets team
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // For compilers that support precompilation, includes "wx/wx.h".
13 #include "wx/wxprec.h"
28 #include "wx/vector.h"
29 #include "wx/module.h"
30 #include "wx/hashmap.h"
31 #include "wx/hashset.h"
33 using namespace wxPrivate
;
35 //-------------------------------------------------------------------------
36 // wxAnyValueTypeGlobals
37 //-------------------------------------------------------------------------
41 WX_DECLARE_HASH_MAP(wxAnyValueType
*,
45 wxAnyTypeToVariantDataFactoryMap
);
49 WX_DECLARE_HASH_SET(wxAnyValueType
*,
52 wxAnyValueTypePtrSet
);
55 // Helper class to manage wxAnyValueType instances and and other
56 // related global variables (such as wxAny<->wxVariant type association).
58 // NB: We really need to have wxAnyValueType instances allocated
59 // in heap. They are stored as static template member variables,
60 // and with them we just can't be too careful (eg. not allocating
61 // them in heap broke the type identification in GCC).
63 class wxAnyValueTypeGlobals
66 wxAnyValueTypeGlobals()
69 ~wxAnyValueTypeGlobals()
72 m_anyToVariant
.clear();
75 wxAnyValueTypePtrSet::iterator it
;
76 for ( it
= m_valueTypes
.begin(); it
!= m_valueTypes
.end(); ++it
)
82 void RegisterValueType(wxAnyValueType
* valueType
)
84 // Let's store value types in set to prevent deleting the same object
85 // several times (it may be possible, under certain conditions, that
86 // the same wxAnyValueType instance gets registered twice)
87 m_valueTypes
.insert(valueType
);
91 void PreRegisterAnyToVariant(wxAnyToVariantRegistration
* reg
)
93 m_anyToVariantRegs
.push_back(reg
);
96 // Find wxVariantData factory function for given value type,
97 // (or compatible, if possible)
98 wxVariantDataFactory
FindVariantDataFactory(const wxAnyValueType
* type_
)
100 // Ideally we'd have the hash map of type 'const wxAnyValueType*',
101 // but WX_DECLARE_HASH_MAP() has some trouble with it.
102 wxAnyValueType
* type
= const_cast<wxAnyValueType
*>(type_
);
104 wxAnyTypeToVariantDataFactoryMap
& anyToVariant
= m_anyToVariant
;
105 wxAnyTypeToVariantDataFactoryMap::const_iterator it
;
106 it
= anyToVariant
.find(type
);
107 if ( it
!= anyToVariant
.end() )
110 // Not found, handle pre-registrations
111 size_t i
= m_anyToVariantRegs
.size();
115 wxAnyToVariantRegistration
* reg
= m_anyToVariantRegs
[i
];
116 wxAnyValueType
* assocType
= reg
->GetAssociatedType();
119 // Both variant data and wxAnyValueType have been
120 // now been properly initialized, so remove the
121 // pre-registration entry and move data to anyToVarian
123 anyToVariant
[assocType
] = reg
->GetFactory();
124 m_anyToVariantRegs
.erase( m_anyToVariantRegs
.begin() + i
);
129 it
= anyToVariant
.find(type
);
130 if ( it
!= anyToVariant
.end() )
133 // Finally, attempt to find a compatible type
134 for ( it
= anyToVariant
.begin(); it
!= anyToVariant
.end(); it
++ )
136 if ( type
->IsSameType(it
->first
) )
138 wxVariantDataFactory f
= it
->second
;
139 anyToVariant
[type
] = f
;
150 wxAnyValueTypePtrSet m_valueTypes
;
152 wxAnyTypeToVariantDataFactoryMap m_anyToVariant
;
153 wxVector
<wxAnyToVariantRegistration
*> m_anyToVariantRegs
;
157 static wxAnyValueTypeGlobals
* g_wxAnyValueTypeGlobals
= NULL
;
161 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplVariantData
)
163 void wxPreRegisterAnyToVariant(wxAnyToVariantRegistration
* reg
)
165 if ( !g_wxAnyValueTypeGlobals
)
166 g_wxAnyValueTypeGlobals
= new wxAnyValueTypeGlobals();
167 g_wxAnyValueTypeGlobals
->PreRegisterAnyToVariant(reg
);
170 bool wxConvertAnyToVariant(const wxAny
& any
, wxVariant
* variant
)
178 // (signed) integer is a special case, because there is only one type
179 // in wxAny, and two ("long" and "longlong") in wxVariant. For better
180 // backwards compatibility, convert all values that fit in "long",
181 // and others to "longlong".
182 if ( wxANY_CHECK_TYPE(any
, signed int) )
186 if ( any
.GetAs(&ll
) )
188 // NB: Do not use LONG_MAX here. Explicitly using 32-bit
189 // integer constraint yields more consistent behavior across
191 if ( ll
> wxINT32_MAX
|| ll
< wxINT32_MIN
)
192 *variant
= wxLongLong(ll
);
194 *variant
= (long) wxLongLong(ll
).GetLo();
210 // Find matching factory function
211 wxVariantDataFactory f
=
212 g_wxAnyValueTypeGlobals
->FindVariantDataFactory(any
.GetType());
214 wxVariantData
* data
= NULL
;
222 // Check if wxAny wrapped wxVariantData*
223 if ( !any
.GetAs(&data
) )
225 // Ok, one last chance: while unlikely, it is possible that the
226 // wxAny actually contains wxVariant.
227 if ( wxANY_CHECK_TYPE(any
, wxVariant
) )
228 *variant
= wxANY_AS(any
, wxVariant
);
232 // Wrapper's GetValue() does not increase reference
233 // count, se have to do it before the data gets passed
238 variant
->SetData(data
);
242 #endif // wxUSE_VARIANT
245 // This class is to make sure that wxAnyValueType instances
246 // etc. get freed correctly. We must use a separate wxAnyValueTypeGlobals
247 // because wxModule itself is instantiated too late.
249 class wxAnyValueTypeGlobalsManager
: public wxModule
251 DECLARE_DYNAMIC_CLASS(wxAnyValueTypeGlobalsManager
)
253 wxAnyValueTypeGlobalsManager() : wxModule() { }
254 virtual ~wxAnyValueTypeGlobalsManager() { }
256 virtual bool OnInit()
260 virtual void OnExit()
262 delete g_wxAnyValueTypeGlobals
;
263 g_wxAnyValueTypeGlobals
= NULL
;
268 IMPLEMENT_DYNAMIC_CLASS(wxAnyValueTypeGlobalsManager
, wxModule
)
271 //-------------------------------------------------------------------------
273 //-------------------------------------------------------------------------
275 wxAnyValueType::wxAnyValueType()
277 if ( !g_wxAnyValueTypeGlobals
)
278 g_wxAnyValueTypeGlobals
= new wxAnyValueTypeGlobals();
280 g_wxAnyValueTypeGlobals
->RegisterValueType(this);
283 //-------------------------------------------------------------------------
284 // Dynamic conversion member functions
285 //-------------------------------------------------------------------------
288 // Define integer minimum and maximum as helpers
290 #define UseIntMin (wxINT64_MIN)
291 #define UseIntMax (wxINT64_MAX)
292 #define UseUintMax (wxUINT64_MAX)
294 #define UseIntMin (LONG_MIN)
295 #define UseIntMax (LONG_MAX)
296 #define UseUintMax (ULONG_MAX)
302 const double UseIntMinF
= static_cast<double>(UseIntMin
);
303 const double UseIntMaxF
= static_cast<double>(UseIntMax
);
304 const double UseUintMaxF
= static_cast<double>(UseUintMax
);
306 } // anonymous namespace
308 bool wxAnyValueTypeImplInt::ConvertValue(const wxAnyValueBuffer
& src
,
309 wxAnyValueType
* dstType
,
310 wxAnyValueBuffer
& dst
) const
312 wxAnyBaseIntType value
= GetValue(src
);
313 if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxString
) )
316 wxLongLong
ll(value
);
317 wxString s
= ll
.ToString();
319 wxString s
= wxString::Format(wxS("%ld"), (long)value
);
321 wxAnyValueTypeImpl
<wxString
>::SetValue(s
, dst
);
323 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseUintType
) )
327 wxAnyBaseUintType ul
= (wxAnyBaseUintType
) value
;
328 wxAnyValueTypeImplUint::SetValue(ul
, dst
);
330 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, double) )
332 double value2
= static_cast<double>(value
);
333 wxAnyValueTypeImplDouble::SetValue(value2
, dst
);
335 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, bool) )
337 bool value2
= value
? true : false;
338 wxAnyValueTypeImpl
<bool>::SetValue(value2
, dst
);
346 bool wxAnyValueTypeImplUint::ConvertValue(const wxAnyValueBuffer
& src
,
347 wxAnyValueType
* dstType
,
348 wxAnyValueBuffer
& dst
) const
350 wxAnyBaseUintType value
= GetValue(src
);
351 if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxString
) )
354 wxULongLong
ull(value
);
355 wxString s
= ull
.ToString();
357 wxString s
= wxString::Format(wxS("%lu"), (long)value
);
359 wxAnyValueTypeImpl
<wxString
>::SetValue(s
, dst
);
361 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseIntType
) )
363 if ( value
> UseIntMax
)
365 wxAnyBaseIntType l
= (wxAnyBaseIntType
) value
;
366 wxAnyValueTypeImplInt::SetValue(l
, dst
);
368 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, double) )
371 double value2
= static_cast<double>(value
);
373 // VC6 doesn't implement conversion from unsigned __int64 to double
374 wxAnyBaseIntType value0
= static_cast<wxAnyBaseIntType
>(value
);
375 double value2
= static_cast<double>(value0
);
377 wxAnyValueTypeImplDouble::SetValue(value2
, dst
);
379 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, bool) )
381 bool value2
= value
? true : false;
382 wxAnyValueTypeImpl
<bool>::SetValue(value2
, dst
);
390 // Convert wxString to destination wxAny value type
391 bool wxAnyConvertString(const wxString
& value
,
392 wxAnyValueType
* dstType
,
393 wxAnyValueBuffer
& dst
)
395 if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxString
) )
397 wxAnyValueTypeImpl
<wxString
>::SetValue(value
, dst
);
399 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseIntType
) )
401 wxAnyBaseIntType value2
;
403 if ( !value
.ToLongLong(&value2
) )
405 if ( !value
.ToLong(&value2
) )
408 wxAnyValueTypeImplInt::SetValue(value2
, dst
);
410 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseUintType
) )
412 wxAnyBaseUintType value2
;
414 if ( !value
.ToULongLong(&value2
) )
416 if ( !value
.ToULong(&value2
) )
419 wxAnyValueTypeImplUint::SetValue(value2
, dst
);
421 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, double) )
424 if ( !value
.ToDouble(&value2
) )
426 wxAnyValueTypeImplDouble::SetValue(value2
, dst
);
428 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, bool) )
433 if ( s
== wxS("true") ||
437 else if ( s
== wxS("false") ||
444 wxAnyValueTypeImpl
<bool>::SetValue(value2
, dst
);
452 bool wxAnyValueTypeImpl
<bool>::ConvertValue(const wxAnyValueBuffer
& src
,
453 wxAnyValueType
* dstType
,
454 wxAnyValueBuffer
& dst
) const
456 bool value
= GetValue(src
);
457 if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseIntType
) )
459 wxAnyBaseIntType value2
= static_cast<wxAnyBaseIntType
>(value
);
460 wxAnyValueTypeImplInt::SetValue(value2
, dst
);
462 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseUintType
) )
464 wxAnyBaseIntType value2
= static_cast<wxAnyBaseUintType
>(value
);
465 wxAnyValueTypeImplUint::SetValue(value2
, dst
);
467 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxString
) )
474 wxAnyValueTypeImpl
<wxString
>::SetValue(s
, dst
);
482 bool wxAnyValueTypeImplDouble::ConvertValue(const wxAnyValueBuffer
& src
,
483 wxAnyValueType
* dstType
,
484 wxAnyValueBuffer
& dst
) const
486 double value
= GetValue(src
);
487 if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseIntType
) )
489 if ( value
< UseIntMinF
|| value
> UseIntMaxF
)
491 wxAnyBaseUintType ul
= static_cast<wxAnyBaseUintType
>(value
);
492 wxAnyValueTypeImplUint::SetValue(ul
, dst
);
494 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxAnyBaseUintType
) )
496 if ( value
< 0.0 || value
> UseUintMaxF
)
498 wxAnyBaseUintType ul
= static_cast<wxAnyBaseUintType
>(value
);
499 wxAnyValueTypeImplUint::SetValue(ul
, dst
);
501 else if ( wxANY_VALUE_TYPE_CHECK_TYPE(dstType
, wxString
) )
503 wxString s
= wxString::Format(wxS("%.14g"), value
);
504 wxAnyValueTypeImpl
<wxString
>::SetValue(s
, dst
);
512 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplInt
)
513 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplUint
)
514 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImpl
<bool>)
515 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplDouble
)
517 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplwxString
)
518 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplConstCharPtr
)
519 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImplConstWchar_tPtr
)
521 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImpl
<wxDateTime
>)
522 //WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImpl<wxObject*>)
523 //WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImpl<wxArrayString>)
525 //-------------------------------------------------------------------------
526 // wxAnyNullValueType implementation
527 //-------------------------------------------------------------------------
536 class wxAnyValueTypeImpl
<wxAnyNullValue
> : public wxAnyValueType
538 WX_DECLARE_ANY_VALUE_TYPE(wxAnyValueTypeImpl
<wxAnyNullValue
>)
540 // Dummy implementations
541 virtual void DeleteValue(wxAnyValueBuffer
& buf
) const
546 virtual void CopyBuffer(const wxAnyValueBuffer
& src
,
547 wxAnyValueBuffer
& dst
) const
553 virtual bool ConvertValue(const wxAnyValueBuffer
& src
,
554 wxAnyValueType
* dstType
,
555 wxAnyValueBuffer
& dst
) const
558 wxUnusedVar(dstType
);
566 WX_IMPLEMENT_ANY_VALUE_TYPE(wxAnyValueTypeImpl
<wxAnyNullValue
>)
568 wxAnyValueType
* wxAnyNullValueType
=
569 wxAnyValueTypeImpl
<wxAnyNullValue
>::GetInstance();