| 
									
										
										
										
											2019-06-29 12:06:26 +02:00
										 |  |  | #include <AK/JsonValue.h>
 | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | #include <LibGUI/GVariant.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-23 07:54:46 +02:00
										 |  |  | const char* to_string(GVariant::Type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case GVariant::Type::Invalid: return "Invalid"; | 
					
						
							|  |  |  |     case GVariant::Type::Bool: return "Bool"; | 
					
						
							|  |  |  |     case GVariant::Type::Int: return "Int"; | 
					
						
							|  |  |  |     case GVariant::Type::Float: return "Float"; | 
					
						
							|  |  |  |     case GVariant::Type::String: return "String"; | 
					
						
							|  |  |  |     case GVariant::Type::Bitmap: return "Bitmap"; | 
					
						
							|  |  |  |     case GVariant::Type::Color: return "Color"; | 
					
						
							|  |  |  |     case GVariant::Type::Icon: return "Icon"; | 
					
						
							|  |  |  |     case GVariant::Type::Point: return "Point"; | 
					
						
							|  |  |  |     case GVariant::Type::Size: return "Size"; | 
					
						
							|  |  |  |     case GVariant::Type::Rect: return "Rect"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  | GVariant::GVariant() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-11 17:28:59 +02:00
										 |  |  |     m_value.as_string = nullptr; | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | GVariant::~GVariant() | 
					
						
							| 
									
										
										
										
											2019-04-12 14:43:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GVariant::clear() | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (m_type) { | 
					
						
							|  |  |  |     case Type::String: | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::deref_if_not_null(m_value.as_string); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Bitmap: | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::deref_if_not_null(m_value.as_bitmap); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  |     case Type::Icon: | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::deref_if_not_null(m_value.as_icon); | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-12 14:43:44 +02:00
										 |  |  |     m_type = Type::Invalid; | 
					
						
							|  |  |  |     m_value.as_string = nullptr; | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GVariant::GVariant(int value) | 
					
						
							|  |  |  |     : m_type(Type::Int) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_int = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GVariant::GVariant(float value) | 
					
						
							|  |  |  |     : m_type(Type::Float) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_float = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GVariant::GVariant(bool value) | 
					
						
							|  |  |  |     : m_type(Type::Bool) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_bool = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 10:37:44 +02:00
										 |  |  | GVariant::GVariant(const char* cstring) | 
					
						
							|  |  |  |     : GVariant(String(cstring)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | GVariant::GVariant(const String& value) | 
					
						
							|  |  |  |     : m_type(Type::String) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_string = const_cast<StringImpl*>(value.impl()); | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |     AK::ref_if_not_null(m_value.as_string); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-29 12:06:26 +02:00
										 |  |  | GVariant::GVariant(const JsonValue& value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (value.is_null()) { | 
					
						
							|  |  |  |         m_value.as_string = nullptr; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_int()) { | 
					
						
							|  |  |  |         m_type = Type::Int; | 
					
						
							|  |  |  |         m_value.as_int = value.as_int(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_uint()) { | 
					
						
							|  |  |  |         ASSERT(value.as_uint() < INT32_MAX); | 
					
						
							|  |  |  |         m_type = Type::Int; | 
					
						
							|  |  |  |         m_value.as_int = value.as_uint(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_string()) { | 
					
						
							|  |  |  |         m_type = Type::String; | 
					
						
							|  |  |  |         m_value.as_string = value.as_string().impl(); | 
					
						
							|  |  |  |         m_value.as_string->ref(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_bool()) { | 
					
						
							|  |  |  |         m_type = Type::Bool; | 
					
						
							|  |  |  |         m_value.as_bool = value.as_bool(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | GVariant::GVariant(const GraphicsBitmap& value) | 
					
						
							|  |  |  |     : m_type(Type::Bitmap) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_bitmap = const_cast<GraphicsBitmap*>(&value); | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |     AK::ref_if_not_null(m_value.as_bitmap); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  | GVariant::GVariant(const GIcon& value) | 
					
						
							|  |  |  |     : m_type(Type::Icon) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_icon = &const_cast<GIconImpl&>(value.impl()); | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |     AK::ref_if_not_null(m_value.as_icon); | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  | GVariant::GVariant(Color color) | 
					
						
							|  |  |  |     : m_type(Type::Color) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_color = color.value(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-11 22:52:34 +02:00
										 |  |  | GVariant::GVariant(const Point& point) | 
					
						
							|  |  |  |     : m_type(Type::Point) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_point = { point.x(), point.y() }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GVariant::GVariant(const Size& size) | 
					
						
							|  |  |  |     : m_type(Type::Size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_size = { size.width(), size.height() }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GVariant::GVariant(const Rect& rect) | 
					
						
							|  |  |  |     : m_type(Type::Rect) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_value.as_rect = (const RawRect&)rect; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-12 14:43:44 +02:00
										 |  |  | GVariant& GVariant::operator=(const GVariant& other) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (&other == this) | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     clear(); | 
					
						
							|  |  |  |     copy_from(other); | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 01:05:59 +02:00
										 |  |  | GVariant& GVariant::operator=(GVariant&& other) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (&other == this) | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     // FIXME: Move, not copy!
 | 
					
						
							|  |  |  |     clear(); | 
					
						
							|  |  |  |     copy_from(other); | 
					
						
							|  |  |  |     other.clear(); | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  | GVariant::GVariant(const GVariant& other) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-12 14:43:44 +02:00
										 |  |  |     copy_from(other); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GVariant::copy_from(const GVariant& other) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(!is_valid()); | 
					
						
							|  |  |  |     m_type = other.m_type; | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  |     switch (m_type) { | 
					
						
							|  |  |  |     case Type::Bool: | 
					
						
							|  |  |  |         m_value.as_bool = other.m_value.as_bool; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Type::Int: | 
					
						
							|  |  |  |         m_value.as_int = other.m_value.as_int; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Type::Float: | 
					
						
							|  |  |  |         m_value.as_float = other.m_value.as_float; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Type::String: | 
					
						
							|  |  |  |         m_value.as_string = other.m_value.as_string; | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::ref_if_not_null(m_value.as_bitmap); | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Bitmap: | 
					
						
							|  |  |  |         m_value.as_bitmap = other.m_value.as_bitmap; | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::ref_if_not_null(m_value.as_bitmap); | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Icon: | 
					
						
							|  |  |  |         m_value.as_icon = other.m_value.as_icon; | 
					
						
							| 
									
										
										
										
											2019-06-21 15:29:31 +02:00
										 |  |  |         AK::ref_if_not_null(m_value.as_icon); | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case Type::Color: | 
					
						
							|  |  |  |         m_value.as_color = other.m_value.as_color; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-04-11 22:52:34 +02:00
										 |  |  |     case Type::Point: | 
					
						
							|  |  |  |         m_value.as_point = other.m_value.as_point; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Type::Size: | 
					
						
							|  |  |  |         m_value.as_size = other.m_value.as_size; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Type::Rect: | 
					
						
							|  |  |  |         m_value.as_rect = other.m_value.as_rect; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-04-11 21:36:04 +02:00
										 |  |  |     case Type::Invalid: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-09 13:33:52 +01:00
										 |  |  | bool GVariant::operator==(const GVariant& other) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_type != other.m_type) | 
					
						
							|  |  |  |         return to_string() == other.to_string(); | 
					
						
							|  |  |  |     switch (m_type) { | 
					
						
							|  |  |  |     case Type::Bool: | 
					
						
							|  |  |  |         return as_bool() == other.as_bool(); | 
					
						
							|  |  |  |     case Type::Int: | 
					
						
							|  |  |  |         return as_int() == other.as_int(); | 
					
						
							|  |  |  |     case Type::Float: | 
					
						
							|  |  |  |         return as_float() == other.as_float(); | 
					
						
							|  |  |  |     case Type::String: | 
					
						
							|  |  |  |         return as_string() == other.as_string(); | 
					
						
							|  |  |  |     case Type::Bitmap: | 
					
						
							|  |  |  |         return m_value.as_bitmap == other.m_value.as_bitmap; | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  |     case Type::Icon: | 
					
						
							|  |  |  |         return m_value.as_icon == other.m_value.as_icon; | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  |     case Type::Color: | 
					
						
							|  |  |  |         return m_value.as_color == other.m_value.as_color; | 
					
						
							| 
									
										
										
										
											2019-04-11 22:52:34 +02:00
										 |  |  |     case Type::Point: | 
					
						
							|  |  |  |         return as_point() == other.as_point(); | 
					
						
							|  |  |  |     case Type::Size: | 
					
						
							|  |  |  |         return as_size() == other.as_size(); | 
					
						
							|  |  |  |     case Type::Rect: | 
					
						
							|  |  |  |         return as_rect() == other.as_rect(); | 
					
						
							| 
									
										
										
										
											2019-03-09 13:33:52 +01:00
										 |  |  |     case Type::Invalid: | 
					
						
							| 
									
										
										
										
											2019-06-29 12:06:26 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2019-03-09 13:33:52 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GVariant::operator<(const GVariant& other) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_type != other.m_type) | 
					
						
							|  |  |  |         return to_string() < other.to_string(); | 
					
						
							|  |  |  |     switch (m_type) { | 
					
						
							|  |  |  |     case Type::Bool: | 
					
						
							|  |  |  |         return as_bool() < other.as_bool(); | 
					
						
							|  |  |  |     case Type::Int: | 
					
						
							|  |  |  |         return as_int() < other.as_int(); | 
					
						
							|  |  |  |     case Type::Float: | 
					
						
							|  |  |  |         return as_float() < other.as_float(); | 
					
						
							|  |  |  |     case Type::String: | 
					
						
							|  |  |  |         return as_string() < other.as_string(); | 
					
						
							|  |  |  |     case Type::Bitmap: | 
					
						
							|  |  |  |         // FIXME: Maybe compare bitmaps somehow differently?
 | 
					
						
							|  |  |  |         return m_value.as_bitmap < other.m_value.as_bitmap; | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  |     case Type::Icon: | 
					
						
							|  |  |  |         // FIXME: Maybe compare icons somehow differently?
 | 
					
						
							|  |  |  |         return m_value.as_icon < other.m_value.as_icon; | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  |     case Type::Color: | 
					
						
							|  |  |  |         return m_value.as_color < other.m_value.as_color; | 
					
						
							| 
									
										
										
										
											2019-04-11 22:52:34 +02:00
										 |  |  |     case Type::Point: | 
					
						
							|  |  |  |     case Type::Size: | 
					
						
							|  |  |  |     case Type::Rect: | 
					
						
							|  |  |  |         // FIXME: Figure out how to compare these.
 | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2019-03-09 13:33:52 +01:00
										 |  |  |     case Type::Invalid: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  | String GVariant::to_string() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (m_type) { | 
					
						
							|  |  |  |     case Type::Bool: | 
					
						
							| 
									
										
										
										
											2019-04-12 14:49:45 +02:00
										 |  |  |         return as_bool() ? "true" : "false"; | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |     case Type::Int: | 
					
						
							|  |  |  |         return String::format("%d", as_int()); | 
					
						
							|  |  |  |     case Type::Float: | 
					
						
							| 
									
										
										
										
											2019-03-06 19:49:59 +01:00
										 |  |  |         return String::format("%f", (double)as_float()); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |     case Type::String: | 
					
						
							|  |  |  |         return as_string(); | 
					
						
							|  |  |  |     case Type::Bitmap: | 
					
						
							|  |  |  |         return "[GraphicsBitmap]"; | 
					
						
							| 
									
										
										
										
											2019-03-24 04:28:36 +01:00
										 |  |  |     case Type::Icon: | 
					
						
							|  |  |  |         return "[GIcon]"; | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  |     case Type::Color: | 
					
						
							|  |  |  |         return as_color().to_string(); | 
					
						
							| 
									
										
										
										
											2019-04-11 22:52:34 +02:00
										 |  |  |     case Type::Point: | 
					
						
							|  |  |  |         return as_point().to_string(); | 
					
						
							|  |  |  |     case Type::Size: | 
					
						
							|  |  |  |         return as_size().to_string(); | 
					
						
							|  |  |  |     case Type::Rect: | 
					
						
							|  |  |  |         return as_rect().to_string(); | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |     case Type::Invalid: | 
					
						
							| 
									
										
										
										
											2019-04-12 14:49:45 +02:00
										 |  |  |         return "[null]"; | 
					
						
							| 
									
										
										
										
											2019-02-28 16:20:29 +01:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } |