Variant.cpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960
  1. // Copyright (c) 2008-2022 the Urho3D project
  2. // License: MIT
  3. #include "../Precompiled.h"
  4. #include "../Core/StringUtils.h"
  5. #include "../IO/VectorBuffer.h"
  6. #include <cstring>
  7. namespace Urho3D
  8. {
  9. const Variant Variant::EMPTY { };
  10. const PODVector<unsigned char> Variant::emptyBuffer { };
  11. const ResourceRef Variant::emptyResourceRef { };
  12. const ResourceRefList Variant::emptyResourceRefList { };
  13. const VariantMap Variant::emptyVariantMap;
  14. const VariantVector Variant::emptyVariantVector { };
  15. const StringVector Variant::emptyStringVector { };
  16. static const char* typeNames[] =
  17. {
  18. "None",
  19. "Int",
  20. "Bool",
  21. "Float",
  22. "Vector2",
  23. "Vector3",
  24. "Vector4",
  25. "Quaternion",
  26. "Color",
  27. "String",
  28. "Buffer",
  29. "VoidPtr",
  30. "ResourceRef",
  31. "ResourceRefList",
  32. "VariantVector",
  33. "VariantMap",
  34. "IntRect",
  35. "IntVector2",
  36. "Ptr",
  37. "Matrix3",
  38. "Matrix3x4",
  39. "Matrix4",
  40. "Double",
  41. "StringVector",
  42. "Rect",
  43. "IntVector3",
  44. "Int64",
  45. "CustomHeap",
  46. "CustomStack",
  47. nullptr
  48. };
  49. static_assert(sizeof(typeNames) / sizeof(const char*) == (size_t)MAX_VAR_TYPES + 1, "Variant type name array is out-of-date");
  50. Variant& Variant::operator =(const Variant& rhs)
  51. {
  52. // Handle custom types separately
  53. if (rhs.IsCustom())
  54. {
  55. SetCustomVariantValue(*rhs.GetCustomVariantValuePtr());
  56. return *this;
  57. }
  58. // Assign other types here
  59. SetType(rhs.GetType());
  60. switch (type_)
  61. {
  62. case VAR_STRING:
  63. value_.string_ = rhs.value_.string_;
  64. break;
  65. case VAR_BUFFER:
  66. value_.buffer_ = rhs.value_.buffer_;
  67. break;
  68. case VAR_RESOURCEREF:
  69. value_.resourceRef_ = rhs.value_.resourceRef_;
  70. break;
  71. case VAR_RESOURCEREFLIST:
  72. value_.resourceRefList_ = rhs.value_.resourceRefList_;
  73. break;
  74. case VAR_VARIANTVECTOR:
  75. value_.variantVector_ = rhs.value_.variantVector_;
  76. break;
  77. case VAR_STRINGVECTOR:
  78. value_.stringVector_ = rhs.value_.stringVector_;
  79. break;
  80. case VAR_VARIANTMAP:
  81. value_.variantMap_ = rhs.value_.variantMap_;
  82. break;
  83. case VAR_PTR:
  84. value_.weakPtr_ = rhs.value_.weakPtr_;
  85. break;
  86. case VAR_MATRIX3:
  87. *value_.matrix3_ = *rhs.value_.matrix3_;
  88. break;
  89. case VAR_MATRIX3X4:
  90. *value_.matrix3x4_ = *rhs.value_.matrix3x4_;
  91. break;
  92. case VAR_MATRIX4:
  93. *value_.matrix4_ = *rhs.value_.matrix4_;
  94. break;
  95. default:
  96. memcpy(&value_, &rhs.value_, sizeof(VariantValue)); // NOLINT(bugprone-undefined-memory-manipulation)
  97. break;
  98. }
  99. return *this;
  100. }
  101. Variant& Variant::operator =(const VectorBuffer& rhs)
  102. {
  103. SetType(VAR_BUFFER);
  104. value_.buffer_ = rhs.GetBuffer();
  105. return *this;
  106. }
  107. bool Variant::operator ==(const Variant& rhs) const
  108. {
  109. if (type_ == VAR_VOIDPTR || type_ == VAR_PTR)
  110. return GetVoidPtr() == rhs.GetVoidPtr();
  111. else if (IsCustom() && rhs.IsCustom())
  112. return GetCustomVariantValuePtr()->Compare(*rhs.GetCustomVariantValuePtr());
  113. else if (type_ != rhs.type_)
  114. return false;
  115. switch (type_)
  116. {
  117. case VAR_INT:
  118. return value_.int_ == rhs.value_.int_;
  119. case VAR_INT64:
  120. return value_.int64_ == rhs.value_.int64_;
  121. case VAR_BOOL:
  122. return value_.bool_ == rhs.value_.bool_;
  123. case VAR_FLOAT:
  124. return value_.float_ == rhs.value_.float_;
  125. case VAR_VECTOR2:
  126. return value_.vector2_ == rhs.value_.vector2_;
  127. case VAR_VECTOR3:
  128. return value_.vector3_ == rhs.value_.vector3_;
  129. case VAR_VECTOR4:
  130. return value_.vector4_ == rhs.value_.vector4_;
  131. case VAR_QUATERNION:
  132. return value_.quaternion_ == rhs.value_.quaternion_;
  133. case VAR_COLOR:
  134. return value_.color_ == rhs.value_.color_;
  135. case VAR_STRING:
  136. return value_.string_ == rhs.value_.string_;
  137. case VAR_BUFFER:
  138. return value_.buffer_ == rhs.value_.buffer_;
  139. case VAR_RESOURCEREF:
  140. return value_.resourceRef_ == rhs.value_.resourceRef_;
  141. case VAR_RESOURCEREFLIST:
  142. return value_.resourceRefList_ == rhs.value_.resourceRefList_;
  143. case VAR_VARIANTVECTOR:
  144. return value_.variantVector_ == rhs.value_.variantVector_;
  145. case VAR_STRINGVECTOR:
  146. return value_.stringVector_ == rhs.value_.stringVector_;
  147. case VAR_VARIANTMAP:
  148. return value_.variantMap_ == rhs.value_.variantMap_;
  149. case VAR_INTRECT:
  150. return value_.intRect_ == rhs.value_.intRect_;
  151. case VAR_INTVECTOR2:
  152. return value_.intVector2_ == rhs.value_.intVector2_;
  153. case VAR_INTVECTOR3:
  154. return value_.intVector3_ == rhs.value_.intVector3_;
  155. case VAR_MATRIX3:
  156. return *value_.matrix3_ == *rhs.value_.matrix3_;
  157. case VAR_MATRIX3X4:
  158. return *value_.matrix3x4_ == *rhs.value_.matrix3x4_;
  159. case VAR_MATRIX4:
  160. return *value_.matrix4_ == *rhs.value_.matrix4_;
  161. case VAR_DOUBLE:
  162. return value_.double_ == rhs.value_.double_;
  163. case VAR_RECT:
  164. return value_.rect_ == rhs.value_.rect_;
  165. default:
  166. return true;
  167. }
  168. }
  169. bool Variant::operator ==(const PODVector<unsigned char>& rhs) const
  170. {
  171. // Use strncmp() instead of PODVector<unsigned char>::operator ==()
  172. const PODVector<unsigned char>& buffer = value_.buffer_;
  173. return type_ == VAR_BUFFER && buffer.Size() == rhs.Size() ?
  174. strncmp(reinterpret_cast<const char*>(&buffer[0]), reinterpret_cast<const char*>(&rhs[0]), buffer.Size()) == 0 :
  175. false;
  176. }
  177. bool Variant::operator ==(const VectorBuffer& rhs) const
  178. {
  179. const PODVector<unsigned char>& buffer = value_.buffer_;
  180. return type_ == VAR_BUFFER && buffer.Size() == rhs.GetSize() ?
  181. strncmp(reinterpret_cast<const char*>(&buffer[0]), reinterpret_cast<const char*>(rhs.GetData()), buffer.Size()) == 0 :
  182. false;
  183. }
  184. void Variant::FromString(const String& type, const String& value)
  185. {
  186. return FromString(GetTypeFromName(type), value.CString());
  187. }
  188. void Variant::FromString(const char* type, const char* value)
  189. {
  190. return FromString(GetTypeFromName(type), value);
  191. }
  192. void Variant::FromString(VariantType type, const String& value)
  193. {
  194. return FromString(type, value.CString());
  195. }
  196. void Variant::FromString(VariantType type, const char* value)
  197. {
  198. switch (type)
  199. {
  200. case VAR_INT:
  201. *this = ToInt(value);
  202. break;
  203. case VAR_INT64:
  204. *this = ToInt64(value);
  205. break;
  206. case VAR_BOOL:
  207. *this = ToBool(value);
  208. break;
  209. case VAR_FLOAT:
  210. *this = ToFloat(value);
  211. break;
  212. case VAR_VECTOR2:
  213. *this = ToVector2(value);
  214. break;
  215. case VAR_VECTOR3:
  216. *this = ToVector3(value);
  217. break;
  218. case VAR_VECTOR4:
  219. *this = ToVector4(value);
  220. break;
  221. case VAR_QUATERNION:
  222. *this = ToQuaternion(value);
  223. break;
  224. case VAR_COLOR:
  225. *this = ToColor(value);
  226. break;
  227. case VAR_STRING:
  228. *this = value;
  229. break;
  230. case VAR_BUFFER:
  231. SetType(VAR_BUFFER);
  232. StringToBuffer(value_.buffer_, value);
  233. break;
  234. case VAR_VOIDPTR:
  235. // From string to void pointer not supported, set to null
  236. *this = (void*)nullptr;
  237. break;
  238. case VAR_RESOURCEREF:
  239. {
  240. StringVector values = String::Split(value, ';');
  241. if (values.Size() == 2)
  242. {
  243. SetType(VAR_RESOURCEREF);
  244. value_.resourceRef_.type_ = values[0];
  245. value_.resourceRef_.name_ = values[1];
  246. }
  247. break;
  248. }
  249. case VAR_RESOURCEREFLIST:
  250. {
  251. StringVector values = String::Split(value, ';', true);
  252. if (values.Size() >= 1)
  253. {
  254. SetType(VAR_RESOURCEREFLIST);
  255. value_.resourceRefList_.type_ = values[0];
  256. value_.resourceRefList_.names_.Resize(values.Size() - 1);
  257. for (unsigned i = 1; i < values.Size(); ++i)
  258. value_.resourceRefList_.names_[i - 1] = values[i];
  259. }
  260. break;
  261. }
  262. case VAR_INTRECT:
  263. *this = ToIntRect(value);
  264. break;
  265. case VAR_INTVECTOR2:
  266. *this = ToIntVector2(value);
  267. break;
  268. case VAR_INTVECTOR3:
  269. *this = ToIntVector3(value);
  270. break;
  271. case VAR_PTR:
  272. // From string to RefCounted pointer not supported, set to null
  273. *this = (RefCounted*)nullptr;
  274. break;
  275. case VAR_MATRIX3:
  276. *this = ToMatrix3(value);
  277. break;
  278. case VAR_MATRIX3X4:
  279. *this = ToMatrix3x4(value);
  280. break;
  281. case VAR_MATRIX4:
  282. *this = ToMatrix4(value);
  283. break;
  284. case VAR_DOUBLE:
  285. *this = ToDouble(value);
  286. break;
  287. case VAR_RECT:
  288. *this = ToRect(value);
  289. break;
  290. default:
  291. SetType(VAR_NONE);
  292. }
  293. }
  294. void Variant::SetBuffer(const void* data, unsigned size)
  295. {
  296. if (size && !data)
  297. size = 0;
  298. SetType(VAR_BUFFER);
  299. PODVector<unsigned char>& buffer = value_.buffer_;
  300. buffer.Resize(size);
  301. if (size)
  302. memcpy(&buffer[0], data, size);
  303. }
  304. void Variant::SetCustomVariantValue(const CustomVariantValue& value)
  305. {
  306. // Assign value if destination is already initialized
  307. if (CustomVariantValue* custom = GetCustomVariantValuePtr())
  308. {
  309. if (custom->GetTypeInfo() == value.GetTypeInfo())
  310. {
  311. custom->Assign(value);
  312. return;
  313. }
  314. }
  315. if (value.GetSize() <= VARIANT_VALUE_SIZE)
  316. {
  317. SetType(VAR_CUSTOM_STACK);
  318. value_.customValueStack_.~CustomVariantValue();
  319. value.Clone(&value_.customValueStack_);
  320. }
  321. else
  322. {
  323. SetType(VAR_CUSTOM_HEAP);
  324. delete value_.customValueHeap_;
  325. value_.customValueHeap_ = value.Clone();
  326. }
  327. }
  328. VectorBuffer Variant::GetVectorBuffer() const
  329. {
  330. return VectorBuffer(type_ == VAR_BUFFER ? value_.buffer_ : emptyBuffer);
  331. }
  332. String Variant::GetTypeName() const
  333. {
  334. return typeNames[type_];
  335. }
  336. String Variant::ToString() const
  337. {
  338. switch (type_)
  339. {
  340. case VAR_INT:
  341. return String(value_.int_);
  342. case VAR_INT64:
  343. return String(value_.int64_);
  344. case VAR_BOOL:
  345. return String(value_.bool_);
  346. case VAR_FLOAT:
  347. return String(value_.float_);
  348. case VAR_VECTOR2:
  349. return value_.vector2_.ToString();
  350. case VAR_VECTOR3:
  351. return value_.vector3_.ToString();
  352. case VAR_VECTOR4:
  353. return value_.vector4_.ToString();
  354. case VAR_QUATERNION:
  355. return value_.quaternion_.ToString();
  356. case VAR_COLOR:
  357. return value_.color_.ToString();
  358. case VAR_STRING:
  359. return value_.string_;
  360. case VAR_BUFFER:
  361. {
  362. const PODVector<unsigned char>& buffer = value_.buffer_;
  363. String ret;
  364. BufferToString(ret, buffer.Begin().ptr_, buffer.Size());
  365. return ret;
  366. }
  367. case VAR_VOIDPTR:
  368. case VAR_PTR:
  369. // Pointer serialization not supported (convert to null)
  370. return String(0);
  371. case VAR_INTRECT:
  372. return value_.intRect_.ToString();
  373. case VAR_INTVECTOR2:
  374. return value_.intVector2_.ToString();
  375. case VAR_INTVECTOR3:
  376. return value_.intVector3_.ToString();
  377. case VAR_MATRIX3:
  378. return value_.matrix3_->ToString();
  379. case VAR_MATRIX3X4:
  380. return value_.matrix3x4_->ToString();
  381. case VAR_MATRIX4:
  382. return value_.matrix4_->ToString();
  383. case VAR_DOUBLE:
  384. return String(value_.double_);
  385. case VAR_RECT:
  386. return value_.rect_.ToString();
  387. case VAR_CUSTOM_HEAP:
  388. case VAR_CUSTOM_STACK:
  389. return GetCustomVariantValuePtr()->ToString();
  390. default:
  391. // VAR_RESOURCEREF, VAR_RESOURCEREFLIST, VAR_VARIANTVECTOR, VAR_STRINGVECTOR, VAR_VARIANTMAP
  392. // Reference string serialization requires typehash-to-name mapping from the context. Can not support here
  393. // Also variant map or vector string serialization is not supported. XML or binary save should be used instead
  394. return String::EMPTY;
  395. }
  396. }
  397. bool Variant::IsZero() const
  398. {
  399. switch (type_)
  400. {
  401. case VAR_INT:
  402. return value_.int_ == 0;
  403. case VAR_INT64:
  404. return value_.int64_ == 0;
  405. case VAR_BOOL:
  406. return value_.bool_ == false;
  407. case VAR_FLOAT:
  408. return value_.float_ == 0.0f;
  409. case VAR_VECTOR2:
  410. return value_.vector2_ == Vector2::ZERO;
  411. case VAR_VECTOR3:
  412. return value_.vector3_ == Vector3::ZERO;
  413. case VAR_VECTOR4:
  414. return value_.vector4_ == Vector4::ZERO;
  415. case VAR_QUATERNION:
  416. return value_.quaternion_ == Quaternion::IDENTITY;
  417. case VAR_COLOR:
  418. // WHITE is considered empty (i.e. default) color in the Color class definition
  419. return value_.color_ == Color::WHITE;
  420. case VAR_STRING:
  421. return value_.string_.Empty();
  422. case VAR_BUFFER:
  423. return value_.buffer_.Empty();
  424. case VAR_VOIDPTR:
  425. return value_.voidPtr_ == nullptr;
  426. case VAR_RESOURCEREF:
  427. return value_.resourceRef_.name_.Empty();
  428. case VAR_RESOURCEREFLIST:
  429. {
  430. const StringVector& names = value_.resourceRefList_.names_;
  431. for (StringVector::ConstIterator i = names.Begin(); i != names.End(); ++i)
  432. {
  433. if (!i->Empty())
  434. return false;
  435. }
  436. return true;
  437. }
  438. case VAR_VARIANTVECTOR:
  439. return value_.variantVector_.Empty();
  440. case VAR_STRINGVECTOR:
  441. return value_.stringVector_.Empty();
  442. case VAR_VARIANTMAP:
  443. return value_.variantMap_.Empty();
  444. case VAR_INTRECT:
  445. return value_.intRect_ == IntRect::ZERO;
  446. case VAR_INTVECTOR2:
  447. return value_.intVector2_ == IntVector2::ZERO;
  448. case VAR_INTVECTOR3:
  449. return value_.intVector3_ == IntVector3::ZERO;
  450. case VAR_PTR:
  451. return value_.weakPtr_ == (RefCounted*)nullptr;
  452. case VAR_MATRIX3:
  453. return *value_.matrix3_ == Matrix3::IDENTITY;
  454. case VAR_MATRIX3X4:
  455. return *value_.matrix3x4_ == Matrix3x4::IDENTITY;
  456. case VAR_MATRIX4:
  457. return *value_.matrix4_ == Matrix4::IDENTITY;
  458. case VAR_DOUBLE:
  459. return value_.double_ == 0.0;
  460. case VAR_RECT:
  461. return value_.rect_ == Rect::ZERO;
  462. case VAR_CUSTOM_HEAP:
  463. case VAR_CUSTOM_STACK:
  464. return GetCustomVariantValuePtr()->IsZero();
  465. default:
  466. return true;
  467. }
  468. }
  469. void Variant::SetType(VariantType newType)
  470. {
  471. if (type_ == newType)
  472. return;
  473. switch (type_)
  474. {
  475. case VAR_STRING:
  476. value_.string_.~String();
  477. break;
  478. case VAR_BUFFER:
  479. value_.buffer_.~PODVector<unsigned char>();
  480. break;
  481. case VAR_RESOURCEREF:
  482. value_.resourceRef_.~ResourceRef();
  483. break;
  484. case VAR_RESOURCEREFLIST:
  485. value_.resourceRefList_.~ResourceRefList();
  486. break;
  487. case VAR_VARIANTVECTOR:
  488. value_.variantVector_.~VariantVector();
  489. break;
  490. case VAR_STRINGVECTOR:
  491. value_.stringVector_.~StringVector();
  492. break;
  493. case VAR_VARIANTMAP:
  494. value_.variantMap_.~VariantMap();
  495. break;
  496. case VAR_PTR:
  497. value_.weakPtr_.~WeakPtr<RefCounted>();
  498. break;
  499. case VAR_MATRIX3:
  500. delete value_.matrix3_;
  501. break;
  502. case VAR_MATRIX3X4:
  503. delete value_.matrix3x4_;
  504. break;
  505. case VAR_MATRIX4:
  506. delete value_.matrix4_;
  507. break;
  508. case VAR_CUSTOM_HEAP:
  509. delete value_.customValueHeap_;
  510. break;
  511. case VAR_CUSTOM_STACK:
  512. value_.customValueStack_.~CustomVariantValue();
  513. break;
  514. default:
  515. break;
  516. }
  517. type_ = newType;
  518. switch (type_)
  519. {
  520. case VAR_STRING:
  521. new(&value_.string_) String();
  522. break;
  523. case VAR_BUFFER:
  524. new(&value_.buffer_) PODVector<unsigned char>();
  525. break;
  526. case VAR_RESOURCEREF:
  527. new(&value_.resourceRef_) ResourceRef();
  528. break;
  529. case VAR_RESOURCEREFLIST:
  530. new(&value_.resourceRefList_) ResourceRefList();
  531. break;
  532. case VAR_VARIANTVECTOR:
  533. new(&value_.variantVector_) VariantVector();
  534. break;
  535. case VAR_STRINGVECTOR:
  536. new(&value_.stringVector_) StringVector();
  537. break;
  538. case VAR_VARIANTMAP:
  539. new(&value_.variantMap_) VariantMap();
  540. break;
  541. case VAR_PTR:
  542. new(&value_.weakPtr_) WeakPtr<RefCounted>();
  543. break;
  544. case VAR_MATRIX3:
  545. value_.matrix3_ = new Matrix3();
  546. break;
  547. case VAR_MATRIX3X4:
  548. value_.matrix3x4_ = new Matrix3x4();
  549. break;
  550. case VAR_MATRIX4:
  551. value_.matrix4_ = new Matrix4();
  552. break;
  553. case VAR_CUSTOM_HEAP:
  554. // Must be initialized later
  555. value_.customValueHeap_ = nullptr;
  556. break;
  557. case VAR_CUSTOM_STACK:
  558. // Initialize virtual table with void custom object
  559. new (&value_.customValueStack_) CustomVariantValue();
  560. break;
  561. default:
  562. break;
  563. }
  564. }
  565. template <> int Variant::Get<int>() const
  566. {
  567. return GetInt();
  568. }
  569. template <> unsigned Variant::Get<unsigned>() const
  570. {
  571. return GetUInt();
  572. }
  573. template <> long long Variant::Get<long long>() const
  574. {
  575. return GetInt64();
  576. }
  577. template <> unsigned long long Variant::Get<unsigned long long>() const
  578. {
  579. return GetUInt64();
  580. }
  581. template <> StringHash Variant::Get<StringHash>() const
  582. {
  583. return GetStringHash();
  584. }
  585. template <> bool Variant::Get<bool>() const
  586. {
  587. return GetBool();
  588. }
  589. template <> float Variant::Get<float>() const
  590. {
  591. return GetFloat();
  592. }
  593. template <> double Variant::Get<double>() const
  594. {
  595. return GetDouble();
  596. }
  597. template <> const Vector2& Variant::Get<const Vector2&>() const
  598. {
  599. return GetVector2();
  600. }
  601. template <> const Vector3& Variant::Get<const Vector3&>() const
  602. {
  603. return GetVector3();
  604. }
  605. template <> const Vector4& Variant::Get<const Vector4&>() const
  606. {
  607. return GetVector4();
  608. }
  609. template <> const Quaternion& Variant::Get<const Quaternion&>() const
  610. {
  611. return GetQuaternion();
  612. }
  613. template <> const Color& Variant::Get<const Color&>() const
  614. {
  615. return GetColor();
  616. }
  617. template <> const String& Variant::Get<const String&>() const
  618. {
  619. return GetString();
  620. }
  621. template <> const Rect& Variant::Get<const Rect&>() const
  622. {
  623. return GetRect();
  624. }
  625. template <> const IntRect& Variant::Get<const IntRect&>() const
  626. {
  627. return GetIntRect();
  628. }
  629. template <> const IntVector2& Variant::Get<const IntVector2&>() const
  630. {
  631. return GetIntVector2();
  632. }
  633. template <> const IntVector3& Variant::Get<const IntVector3&>() const
  634. {
  635. return GetIntVector3();
  636. }
  637. template <> const PODVector<unsigned char>& Variant::Get<const PODVector<unsigned char>&>() const
  638. {
  639. return GetBuffer();
  640. }
  641. template <> void* Variant::Get<void*>() const
  642. {
  643. return GetVoidPtr();
  644. }
  645. template <> RefCounted* Variant::Get<RefCounted*>() const
  646. {
  647. return GetPtr();
  648. }
  649. template <> const Matrix3& Variant::Get<const Matrix3&>() const
  650. {
  651. return GetMatrix3();
  652. }
  653. template <> const Matrix3x4& Variant::Get<const Matrix3x4&>() const
  654. {
  655. return GetMatrix3x4();
  656. }
  657. template <> const Matrix4& Variant::Get<const Matrix4&>() const
  658. {
  659. return GetMatrix4();
  660. }
  661. template <> ResourceRef Variant::Get<ResourceRef>() const
  662. {
  663. return GetResourceRef();
  664. }
  665. template <> ResourceRefList Variant::Get<ResourceRefList>() const
  666. {
  667. return GetResourceRefList();
  668. }
  669. template <> VariantVector Variant::Get<VariantVector>() const
  670. {
  671. return GetVariantVector();
  672. }
  673. template <> StringVector Variant::Get<StringVector >() const
  674. {
  675. return GetStringVector();
  676. }
  677. template <> VariantMap Variant::Get<VariantMap>() const
  678. {
  679. return GetVariantMap();
  680. }
  681. template <> Vector2 Variant::Get<Vector2>() const
  682. {
  683. return GetVector2();
  684. }
  685. template <> Vector3 Variant::Get<Vector3>() const
  686. {
  687. return GetVector3();
  688. }
  689. template <> Vector4 Variant::Get<Vector4>() const
  690. {
  691. return GetVector4();
  692. }
  693. template <> Quaternion Variant::Get<Quaternion>() const
  694. {
  695. return GetQuaternion();
  696. }
  697. template <> Color Variant::Get<Color>() const
  698. {
  699. return GetColor();
  700. }
  701. template <> String Variant::Get<String>() const
  702. {
  703. return GetString();
  704. }
  705. template <> Rect Variant::Get<Rect>() const
  706. {
  707. return GetRect();
  708. }
  709. template <> IntRect Variant::Get<IntRect>() const
  710. {
  711. return GetIntRect();
  712. }
  713. template <> IntVector2 Variant::Get<IntVector2>() const
  714. {
  715. return GetIntVector2();
  716. }
  717. template <> IntVector3 Variant::Get<IntVector3>() const
  718. {
  719. return GetIntVector3();
  720. }
  721. template <> PODVector<unsigned char> Variant::Get<PODVector<unsigned char>>() const
  722. {
  723. return GetBuffer();
  724. }
  725. template <> Matrix3 Variant::Get<Matrix3>() const
  726. {
  727. return GetMatrix3();
  728. }
  729. template <> Matrix3x4 Variant::Get<Matrix3x4>() const
  730. {
  731. return GetMatrix3x4();
  732. }
  733. template <> Matrix4 Variant::Get<Matrix4>() const
  734. {
  735. return GetMatrix4();
  736. }
  737. String Variant::GetTypeName(VariantType type)
  738. {
  739. return typeNames[type];
  740. }
  741. VariantType Variant::GetTypeFromName(const String& typeName)
  742. {
  743. return GetTypeFromName(typeName.CString());
  744. }
  745. VariantType Variant::GetTypeFromName(const char* typeName)
  746. {
  747. return (VariantType)GetStringListIndex(typeName, typeNames, VAR_NONE);
  748. }
  749. }