mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 00:52:33 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			353 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			353 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2022, Filiph Sandström <filiph.sandstrom@filfatstudios.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/String.h>
 | |
| #include <LibGUI/Icon.h>
 | |
| #include <LibGfx/Bitmap.h>
 | |
| #include <LibGfx/Font.h>
 | |
| #include <LibGfx/SystemTheme.h>
 | |
| 
 | |
| namespace GUI {
 | |
| 
 | |
| class Variant {
 | |
| public:
 | |
|     Variant();
 | |
|     Variant(bool);
 | |
|     Variant(float);
 | |
|     Variant(i32);
 | |
|     Variant(i64);
 | |
|     Variant(u32);
 | |
|     Variant(u64);
 | |
|     Variant(char const*);
 | |
|     Variant(StringView);
 | |
|     Variant(String const&);
 | |
|     Variant(FlyString const&);
 | |
|     Variant(Gfx::Bitmap const&);
 | |
|     Variant(const GUI::Icon&);
 | |
|     Variant(Gfx::IntPoint const&);
 | |
|     Variant(Gfx::IntSize const&);
 | |
|     Variant(Gfx::IntRect const&);
 | |
|     Variant(Gfx::Font const&);
 | |
|     Variant(const Gfx::TextAlignment);
 | |
|     Variant(const Gfx::ColorRole);
 | |
|     Variant(const Gfx::AlignmentRole);
 | |
|     Variant(const Gfx::FlagRole);
 | |
|     Variant(const Gfx::MetricRole);
 | |
|     Variant(const Gfx::PathRole);
 | |
|     Variant(JsonValue const&);
 | |
|     Variant(Color);
 | |
| 
 | |
|     Variant(Variant const&);
 | |
|     Variant& operator=(Variant const&);
 | |
| 
 | |
|     Variant(Variant&&) = delete;
 | |
|     Variant& operator=(Variant&&);
 | |
| 
 | |
|     void clear();
 | |
|     ~Variant();
 | |
| 
 | |
|     enum class Type {
 | |
|         Invalid,
 | |
|         Bool,
 | |
|         Int32,
 | |
|         Int64,
 | |
|         UnsignedInt32,
 | |
|         UnsignedInt64,
 | |
|         Float,
 | |
|         String,
 | |
|         Bitmap,
 | |
|         Color,
 | |
|         Icon,
 | |
|         Point,
 | |
|         Size,
 | |
|         Rect,
 | |
|         Font,
 | |
|         TextAlignment,
 | |
|         ColorRole,
 | |
|         AlignmentRole,
 | |
|         FlagRole,
 | |
|         MetricRole,
 | |
|         PathRole,
 | |
|     };
 | |
| 
 | |
|     bool is_valid() const { return m_type != Type::Invalid; }
 | |
|     bool is_bool() const { return m_type == Type::Bool; }
 | |
|     bool is_i32() const { return m_type == Type::Int32; }
 | |
|     bool is_i64() const { return m_type == Type::Int64; }
 | |
|     bool is_u32() const { return m_type == Type::UnsignedInt32; }
 | |
|     bool is_u64() const { return m_type == Type::UnsignedInt64; }
 | |
|     bool is_float() const { return m_type == Type::Float; }
 | |
|     bool is_string() const { return m_type == Type::String; }
 | |
|     bool is_bitmap() const { return m_type == Type::Bitmap; }
 | |
|     bool is_color() const { return m_type == Type::Color; }
 | |
|     bool is_icon() const { return m_type == Type::Icon; }
 | |
|     bool is_point() const { return m_type == Type::Point; }
 | |
|     bool is_size() const { return m_type == Type::Size; }
 | |
|     bool is_rect() const { return m_type == Type::Rect; }
 | |
|     bool is_font() const { return m_type == Type::Font; }
 | |
|     bool is_text_alignment() const { return m_type == Type::TextAlignment; }
 | |
|     bool is_color_role() const { return m_type == Type::ColorRole; }
 | |
|     bool is_alignment_role() const { return m_type == Type::AlignmentRole; }
 | |
|     bool is_flag_role() const { return m_type == Type::FlagRole; }
 | |
|     bool is_metric_role() const { return m_type == Type::MetricRole; }
 | |
|     bool is_path_role() const { return m_type == Type::PathRole; }
 | |
|     Type type() const { return m_type; }
 | |
| 
 | |
|     bool as_bool() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Bool);
 | |
|         return m_value.as_bool;
 | |
|     }
 | |
| 
 | |
|     bool to_bool() const
 | |
|     {
 | |
|         if (type() == Type::Bool)
 | |
|             return as_bool();
 | |
|         if (type() == Type::String)
 | |
|             return !!m_value.as_string;
 | |
|         if (type() == Type::Int32)
 | |
|             return m_value.as_i32 != 0;
 | |
|         if (type() == Type::Int64)
 | |
|             return m_value.as_i64 != 0;
 | |
|         if (type() == Type::UnsignedInt32)
 | |
|             return m_value.as_u32 != 0;
 | |
|         if (type() == Type::UnsignedInt64)
 | |
|             return m_value.as_u64 != 0;
 | |
|         if (type() == Type::Rect)
 | |
|             return !as_rect().is_null();
 | |
|         if (type() == Type::Size)
 | |
|             return !as_size().is_null();
 | |
|         if (type() == Type::Point)
 | |
|             return !as_point().is_null();
 | |
|         return is_valid();
 | |
|     }
 | |
| 
 | |
|     int as_i32() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Int32);
 | |
|         return m_value.as_i32;
 | |
|     }
 | |
| 
 | |
|     int as_i64() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Int64);
 | |
|         return m_value.as_i64;
 | |
|     }
 | |
| 
 | |
|     u32 as_u32() const
 | |
|     {
 | |
|         VERIFY(type() == Type::UnsignedInt32);
 | |
|         return m_value.as_u32;
 | |
|     }
 | |
| 
 | |
|     u64 as_u64() const
 | |
|     {
 | |
|         VERIFY(type() == Type::UnsignedInt64);
 | |
|         return m_value.as_u64;
 | |
|     }
 | |
| 
 | |
|     template<typename T>
 | |
|     T to_integer() const
 | |
|     {
 | |
|         if (is_i32())
 | |
|             return as_i32();
 | |
|         if (is_i64())
 | |
|             return as_i64();
 | |
|         if (is_bool())
 | |
|             return as_bool() ? 1 : 0;
 | |
|         if (is_float())
 | |
|             return (int)as_float();
 | |
|         if (is_u32()) {
 | |
|             VERIFY(as_u32() <= INT32_MAX);
 | |
|             return static_cast<i32>(as_u32());
 | |
|         }
 | |
|         if (is_u64()) {
 | |
|             VERIFY(as_u64() <= INT64_MAX);
 | |
|             return static_cast<i64>(as_u64());
 | |
|         }
 | |
|         if (is_string())
 | |
|             return as_string().to_int().value_or(0);
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     i32 to_i32() const
 | |
|     {
 | |
|         return to_integer<i32>();
 | |
|     }
 | |
| 
 | |
|     i64 to_i64() const
 | |
|     {
 | |
|         return to_integer<i64>();
 | |
|     }
 | |
| 
 | |
|     float as_float() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Float);
 | |
|         return m_value.as_float;
 | |
|     }
 | |
| 
 | |
|     float as_float_or(float fallback) const
 | |
|     {
 | |
|         if (is_float())
 | |
|             return as_float();
 | |
|         return fallback;
 | |
|     }
 | |
| 
 | |
|     Gfx::IntPoint as_point() const
 | |
|     {
 | |
|         return { m_value.as_point.x, m_value.as_point.y };
 | |
|     }
 | |
| 
 | |
|     Gfx::IntSize as_size() const
 | |
|     {
 | |
|         return { m_value.as_size.width, m_value.as_size.height };
 | |
|     }
 | |
| 
 | |
|     Gfx::IntRect as_rect() const
 | |
|     {
 | |
|         return { as_point(), as_size() };
 | |
|     }
 | |
| 
 | |
|     String as_string() const
 | |
|     {
 | |
|         VERIFY(type() == Type::String);
 | |
|         return m_value.as_string;
 | |
|     }
 | |
| 
 | |
|     Gfx::Bitmap const& as_bitmap() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Bitmap);
 | |
|         return *m_value.as_bitmap;
 | |
|     }
 | |
| 
 | |
|     GUI::Icon as_icon() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Icon);
 | |
|         return GUI::Icon(*m_value.as_icon);
 | |
|     }
 | |
| 
 | |
|     Color as_color() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Color);
 | |
|         return Color::from_argb(m_value.as_color);
 | |
|     }
 | |
| 
 | |
|     Gfx::Font const& as_font() const
 | |
|     {
 | |
|         VERIFY(type() == Type::Font);
 | |
|         return *m_value.as_font;
 | |
|     }
 | |
| 
 | |
|     Gfx::TextAlignment to_text_alignment(Gfx::TextAlignment default_value) const
 | |
|     {
 | |
|         if (type() != Type::TextAlignment)
 | |
|             return default_value;
 | |
|         return m_value.as_text_alignment;
 | |
|     }
 | |
| 
 | |
|     Gfx::ColorRole to_color_role() const
 | |
|     {
 | |
|         if (type() != Type::ColorRole)
 | |
|             return Gfx::ColorRole::NoRole;
 | |
|         return m_value.as_color_role;
 | |
|     }
 | |
| 
 | |
|     Gfx::AlignmentRole to_alignment_role() const
 | |
|     {
 | |
|         if (type() != Type::AlignmentRole)
 | |
|             return Gfx::AlignmentRole::NoRole;
 | |
|         return m_value.as_alignment_role;
 | |
|     }
 | |
| 
 | |
|     Gfx::FlagRole to_flag_role() const
 | |
|     {
 | |
|         if (type() != Type::FlagRole)
 | |
|             return Gfx::FlagRole::NoRole;
 | |
|         return m_value.as_flag_role;
 | |
|     }
 | |
| 
 | |
|     Gfx::MetricRole to_metric_role() const
 | |
|     {
 | |
|         if (type() != Type::MetricRole)
 | |
|             return Gfx::MetricRole::NoRole;
 | |
|         return m_value.as_metric_role;
 | |
|     }
 | |
| 
 | |
|     Gfx::PathRole to_path_role() const
 | |
|     {
 | |
|         if (type() != Type::PathRole)
 | |
|             return Gfx::PathRole::NoRole;
 | |
|         return m_value.as_path_role;
 | |
|     }
 | |
| 
 | |
|     Color to_color(Color default_value = {}) const
 | |
|     {
 | |
|         if (type() == Type::Color)
 | |
|             return as_color();
 | |
|         if (type() == Type::String) {
 | |
|             auto color = Color::from_string(as_string());
 | |
|             if (color.has_value())
 | |
|                 return color.value();
 | |
|         }
 | |
|         return default_value;
 | |
|     }
 | |
| 
 | |
|     String to_string() const;
 | |
| 
 | |
|     bool operator==(Variant const&) const;
 | |
|     bool operator<(Variant const&) const;
 | |
| 
 | |
| private:
 | |
|     void copy_from(Variant const&);
 | |
|     void move_from(Variant&&);
 | |
| 
 | |
|     struct RawPoint {
 | |
|         int x;
 | |
|         int y;
 | |
|     };
 | |
| 
 | |
|     struct RawSize {
 | |
|         int width;
 | |
|         int height;
 | |
|     };
 | |
| 
 | |
|     struct RawRect {
 | |
|         RawPoint location;
 | |
|         RawSize size;
 | |
|     };
 | |
| 
 | |
|     union {
 | |
|         StringImpl* as_string;
 | |
|         Gfx::Bitmap* as_bitmap;
 | |
|         GUI::IconImpl* as_icon;
 | |
|         Gfx::Font* as_font;
 | |
|         bool as_bool;
 | |
|         i32 as_i32;
 | |
|         i64 as_i64;
 | |
|         u32 as_u32;
 | |
|         u64 as_u64;
 | |
|         float as_float;
 | |
|         Gfx::ARGB32 as_color;
 | |
|         Gfx::TextAlignment as_text_alignment;
 | |
|         Gfx::ColorRole as_color_role;
 | |
|         Gfx::AlignmentRole as_alignment_role;
 | |
|         Gfx::FlagRole as_flag_role;
 | |
|         Gfx::MetricRole as_metric_role;
 | |
|         Gfx::PathRole as_path_role;
 | |
|         RawPoint as_point;
 | |
|         RawSize as_size;
 | |
|         RawRect as_rect;
 | |
|     } m_value;
 | |
| 
 | |
|     Type m_type { Type::Invalid };
 | |
| };
 | |
| 
 | |
| char const* to_string(Variant::Type);
 | |
| 
 | |
| }
 | 
