| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 |
- // Copyright Benoit Blanchon 2014-2016
- // MIT License
- //
- // Arduino JSON library
- // https://github.com/bblanchon/ArduinoJson
- // If you like this project, please add a star!
- #pragma once
- #include <stddef.h>
- #include <stdint.h> // for uint8_t
- #include "Internals/JsonPrintable.hpp"
- #include "Internals/JsonVariantContent.hpp"
- #include "Internals/JsonVariantType.hpp"
- #include "JsonVariantBase.hpp"
- #include "RawJson.hpp"
- #include "TypeTraits/EnableIf.hpp"
- #include "TypeTraits/IsFloatingPoint.hpp"
- #include "TypeTraits/IsIntegral.hpp"
- #include "TypeTraits/IsSame.hpp"
- #include "TypeTraits/RemoveConst.hpp"
- #include "TypeTraits/RemoveReference.hpp"
- namespace ArduinoJson {
- // Forward declarations.
- class JsonArray;
- class JsonObject;
- // A variant that can be a any value serializable to a JSON value.
- //
- // It can be set to:
- // - a boolean
- // - a char, short, int or a long (signed or unsigned)
- // - a string (const char*)
- // - a reference to a JsonArray or JsonObject
- class JsonVariant : public JsonVariantBase<JsonVariant> {
- public:
- template <typename T>
- struct IsConstructibleFrom;
- // Creates an uninitialized JsonVariant
- JsonVariant() : _type(Internals::JSON_UNDEFINED) {}
- // Create a JsonVariant containing a boolean value.
- // It will be serialized as "true" or "false" in JSON.
- JsonVariant(bool value) {
- using namespace Internals;
- _type = JSON_BOOLEAN;
- _content.asInteger = static_cast<JsonInteger>(value);
- }
- // Create a JsonVariant containing a floating point value.
- // The second argument specifies the number of decimal digits to write in
- // the JSON string.
- // JsonVariant(double value, uint8_t decimals);
- // JsonVariant(float value, uint8_t decimals);
- template <typename T>
- JsonVariant(T value, uint8_t decimals = 2,
- typename TypeTraits::EnableIf<
- TypeTraits::IsFloatingPoint<T>::value>::type * = 0) {
- using namespace Internals;
- _type = static_cast<JsonVariantType>(JSON_FLOAT_0_DECIMALS + decimals);
- _content.asFloat = static_cast<JsonFloat>(value);
- }
- // Create a JsonVariant containing an integer value.
- // JsonVariant(signed short)
- // JsonVariant(signed int)
- // JsonVariant(signed long)
- template <typename T>
- JsonVariant(T value,
- typename TypeTraits::EnableIf<
- TypeTraits::IsSignedIntegral<T>::value>::type * = 0) {
- using namespace Internals;
- if (value >= 0) {
- _type = JSON_POSITIVE_INTEGER;
- _content.asInteger = static_cast<JsonUInt>(value);
- } else {
- _type = JSON_NEGATIVE_INTEGER;
- _content.asInteger = static_cast<JsonUInt>(-value);
- }
- }
- // JsonVariant(unsigned short)
- // JsonVariant(unsigned int)
- // JsonVariant(unsigned long)
- template <typename T>
- JsonVariant(T value,
- typename TypeTraits::EnableIf<
- TypeTraits::IsUnsignedIntegral<T>::value>::type * = 0) {
- using namespace Internals;
- _type = JSON_POSITIVE_INTEGER;
- _content.asInteger = static_cast<JsonUInt>(value);
- }
- // Create a JsonVariant containing a string.
- JsonVariant(const char *value) {
- _type = Internals::JSON_STRING;
- _content.asString = value;
- }
- // Create a JsonVariant containing an unparsed string
- JsonVariant(RawJson value) {
- _type = Internals::JSON_UNPARSED;
- _content.asString = value;
- }
- // Create a JsonVariant containing a reference to an array.
- JsonVariant(JsonArray &array) {
- _type = Internals::JSON_ARRAY;
- _content.asArray = &array;
- }
- // Create a JsonVariant containing a reference to an object.
- JsonVariant(JsonObject &object) {
- _type = Internals::JSON_OBJECT;
- _content.asObject = &object;
- }
- // Get the variant as the specified type.
- //
- // short as<signed short>() const;
- // int as<signed int>() const;
- // long as<signed long>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsSignedIntegral<T>::value,
- T>::type
- as() const {
- return static_cast<T>(asInteger());
- }
- //
- // short as<unsigned short>() const;
- // int as<unsigned int>() const;
- // long as<unsigned long>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsUnsignedIntegral<T>::value,
- T>::type
- as() const {
- return static_cast<T>(asUnsignedInteger());
- }
- //
- // double as<double>() const;
- // float as<float>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<T>::value,
- T>::type
- as() const {
- return static_cast<T>(asFloat());
- }
- //
- // const String as<String>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsSame<T, String>::value,
- T>::type
- as() const {
- return toString();
- }
- //
- // const char* as<const char*>() const;
- // const char* as<char*>() const;
- template <typename T>
- typename TypeTraits::EnableIf<TypeTraits::IsSame<T, const char *>::value ||
- TypeTraits::IsSame<T, char *>::value,
- const char *>::type
- as() const {
- return asString();
- }
- //
- // const bool as<bool>() const
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsSame<T, bool>::value,
- T>::type
- as() const {
- return asInteger() != 0;
- }
- //
- // JsonArray& as<JsonArray> const;
- // JsonArray& as<JsonArray&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type,
- JsonArray>::value,
- JsonArray &>::type
- as() const {
- return asArray();
- }
- //
- // const JsonArray& as<const JsonArray&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type,
- const JsonArray>::value,
- const JsonArray &>::type
- as() const {
- return asArray();
- }
- //
- // JsonObject& as<JsonObject> const;
- // JsonObject& as<JsonObject&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type,
- JsonObject>::value,
- JsonObject &>::type
- as() const {
- return asObject();
- }
- //
- // JsonObject& as<const JsonObject> const;
- // JsonObject& as<const JsonObject&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<typename TypeTraits::RemoveReference<T>::type,
- const JsonObject>::value,
- const JsonObject &>::type
- as() const {
- return asObject();
- }
- // Tells weither the variant has the specified type.
- // Returns true if the variant has type type T, false otherwise.
- //
- // short as<short>() const;
- // int as<int>() const;
- // long as<long>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsIntegral<T>::value,
- bool>::type
- is() const {
- return isInteger();
- }
- //
- // double is<double>() const;
- // float is<float>() const;
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsFloatingPoint<T>::value,
- bool>::type
- is() const {
- return isFloat();
- }
- //
- // const bool is<bool>() const
- template <typename T>
- const typename TypeTraits::EnableIf<TypeTraits::IsSame<T, bool>::value,
- bool>::type
- is() const {
- return isBoolean();
- }
- //
- // bool is<const char*>() const;
- // bool is<char*>() const;
- template <typename T>
- typename TypeTraits::EnableIf<TypeTraits::IsSame<T, const char *>::value ||
- TypeTraits::IsSame<T, char *>::value,
- bool>::type
- is() const {
- return isString();
- }
- //
- // bool is<JsonArray> const;
- // bool is<JsonArray&> const;
- // bool is<const JsonArray&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<
- typename TypeTraits::RemoveConst<
- typename TypeTraits::RemoveReference<T>::type>::type,
- JsonArray>::value,
- bool>::type
- is() const {
- return isArray();
- }
- //
- // JsonObject& as<JsonObject> const;
- // JsonObject& as<JsonObject&> const;
- // JsonObject& as<const JsonObject&> const;
- template <typename T>
- typename TypeTraits::EnableIf<
- TypeTraits::IsSame<
- typename TypeTraits::RemoveConst<
- typename TypeTraits::RemoveReference<T>::type>::type,
- JsonObject>::value,
- bool>::type
- is() const {
- return isObject();
- }
- // Returns true if the variant has a value
- bool success() const {
- return _type != Internals::JSON_UNDEFINED;
- }
- // Serialize the variant to a JsonWriter
- void writeTo(Internals::JsonWriter &writer) const;
- // Value returned if the variant has an incompatible type
- template <typename T>
- static typename Internals::JsonVariantAs<T>::type defaultValue() {
- return T();
- }
- const char *asString() const;
- JsonArray &asArray() const;
- JsonObject &asObject() const;
- private:
- // It's not allowed to store a char
- template <typename T>
- JsonVariant(T value, typename TypeTraits::EnableIf<
- TypeTraits::IsSame<T, char>::value>::type * = 0);
- String toString() const;
- Internals::JsonFloat asFloat() const;
- Internals::JsonInteger asInteger() const;
- Internals::JsonUInt asUnsignedInteger() const;
- bool isBoolean() const;
- bool isFloat() const;
- bool isInteger() const;
- bool isArray() const {
- return _type == Internals::JSON_ARRAY;
- }
- bool isObject() const {
- return _type == Internals::JSON_OBJECT;
- }
- bool isString() const {
- return _type == Internals::JSON_STRING;
- }
- // The current type of the variant
- Internals::JsonVariantType _type;
- // The various alternatives for the value of the variant.
- Internals::JsonVariantContent _content;
- };
- inline JsonVariant float_with_n_digits(float value, uint8_t digits) {
- return JsonVariant(value, digits);
- }
- inline JsonVariant double_with_n_digits(double value, uint8_t digits) {
- return JsonVariant(value, digits);
- }
- template <typename T>
- struct JsonVariant::IsConstructibleFrom {
- static const bool value =
- TypeTraits::IsIntegral<T>::value ||
- TypeTraits::IsFloatingPoint<T>::value ||
- TypeTraits::IsSame<T, bool>::value ||
- TypeTraits::IsSame<T, char *>::value ||
- TypeTraits::IsSame<T, const char *>::value ||
- TypeTraits::IsSame<T, RawJson>::value ||
- TypeTraits::IsSame<T, JsonArray &>::value ||
- TypeTraits::IsSame<T, const JsonArray &>::value ||
- TypeTraits::IsSame<T, JsonArraySubscript &>::value ||
- TypeTraits::IsSame<T, const JsonArraySubscript &>::value ||
- TypeTraits::IsSame<T, JsonObject &>::value ||
- TypeTraits::IsSame<T, const JsonObject &>::value ||
- TypeTraits::IsSame<T, JsonObjectSubscript<const char *> &>::value ||
- TypeTraits::IsSame<T, const JsonObjectSubscript<const char *> &>::value ||
- TypeTraits::IsSame<T, JsonObjectSubscript<String> &>::value ||
- TypeTraits::IsSame<T, const JsonObjectSubscript<String> &>::value ||
- TypeTraits::IsSame<T, JsonVariant &>::value ||
- TypeTraits::IsSame<T, const JsonVariant &>::value;
- };
- }
|