| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2023, MacDue <macdue@dueutil.tech> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <AK/Array.h>
 | 
					
						
							|  |  |  | #include <AK/Endian.h>
 | 
					
						
							|  |  |  | #include <AK/FixedArray.h>
 | 
					
						
							|  |  |  | #include <AK/LEB128.h>
 | 
					
						
							|  |  |  | #include <AK/MemoryStream.h>
 | 
					
						
							|  |  |  | #include <AK/Variant.h>
 | 
					
						
							|  |  |  | #include <LibCore/File.h>
 | 
					
						
							|  |  |  | #include <LibGfx/AntiAliasingPainter.h>
 | 
					
						
							|  |  |  | #include <LibGfx/ImageFormats/TinyVGLoader.h>
 | 
					
						
							|  |  |  | #include <LibGfx/Line.h>
 | 
					
						
							|  |  |  | #include <LibGfx/Painter.h>
 | 
					
						
							|  |  |  | #include <LibGfx/Point.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Gfx { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using VarUInt = LEB128<u32>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static constexpr Array<u8, 2> TVG_MAGIC { 0x72, 0x56 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class ColorEncoding : u8 { | 
					
						
							|  |  |  |     RGBA8888 = 0, | 
					
						
							|  |  |  |     RGB565 = 1, | 
					
						
							|  |  |  |     RGBAF32 = 2, | 
					
						
							|  |  |  |     Custom = 3 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class CoordinateRange : u8 { | 
					
						
							|  |  |  |     Default = 0, | 
					
						
							|  |  |  |     Reduced = 1, | 
					
						
							|  |  |  |     Enhanced = 2 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class StyleType : u8 { | 
					
						
							|  |  |  |     FlatColored = 0, | 
					
						
							|  |  |  |     LinearGradient = 1, | 
					
						
							|  |  |  |     RadialGradinet = 2 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class Command : u8 { | 
					
						
							|  |  |  |     EndOfDocument = 0, | 
					
						
							|  |  |  |     FillPolygon = 1, | 
					
						
							|  |  |  |     FillRectangles = 2, | 
					
						
							|  |  |  |     FillPath = 3, | 
					
						
							|  |  |  |     DrawLines = 4, | 
					
						
							|  |  |  |     DrawLineLoop = 5, | 
					
						
							|  |  |  |     DrawLineStrip = 6, | 
					
						
							|  |  |  |     DrawLinePath = 7, | 
					
						
							|  |  |  |     OutlineFillPolygon = 8, | 
					
						
							|  |  |  |     OutlineFillRectangles = 9, | 
					
						
							|  |  |  |     OutLineFillPath = 10 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct FillCommandHeader { | 
					
						
							|  |  |  |     u32 count; | 
					
						
							|  |  |  |     TinyVGDecodedImageData::Style style; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct DrawCommandHeader { | 
					
						
							|  |  |  |     u32 count; | 
					
						
							|  |  |  |     TinyVGDecodedImageData::Style line_style; | 
					
						
							|  |  |  |     float line_width; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct OutlineFillCommandHeader { | 
					
						
							|  |  |  |     u32 count; | 
					
						
							|  |  |  |     TinyVGDecodedImageData::Style fill_style; | 
					
						
							|  |  |  |     TinyVGDecodedImageData::Style line_style; | 
					
						
							|  |  |  |     float line_width; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class PathCommand : u8 { | 
					
						
							|  |  |  |     Line = 0, | 
					
						
							|  |  |  |     HorizontalLine = 1, | 
					
						
							|  |  |  |     VerticalLine = 2, | 
					
						
							|  |  |  |     CubicBezier = 3, | 
					
						
							|  |  |  |     ArcCircle = 4, | 
					
						
							|  |  |  |     ArcEllipse = 5, | 
					
						
							|  |  |  |     ClosePath = 6, | 
					
						
							|  |  |  |     QuadraticBezier = 7 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct TinyVGHeader { | 
					
						
							|  |  |  |     u8 version; | 
					
						
							|  |  |  |     u8 scale; | 
					
						
							|  |  |  |     ColorEncoding color_encoding; | 
					
						
							|  |  |  |     CoordinateRange coordinate_range; | 
					
						
							|  |  |  |     u32 width; | 
					
						
							|  |  |  |     u32 height; | 
					
						
							|  |  |  |     u32 color_count; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ErrorOr<TinyVGHeader> decode_tinyvg_header(Stream& stream) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     TinyVGHeader header {}; | 
					
						
							|  |  |  |     Array<u8, 2> magic_bytes; | 
					
						
							|  |  |  |     TRY(stream.read_until_filled(magic_bytes)); | 
					
						
							|  |  |  |     if (magic_bytes != TVG_MAGIC) | 
					
						
							|  |  |  |         return Error::from_string_literal("Invalid TinyVG: Incorrect header magic"); | 
					
						
							|  |  |  |     header.version = TRY(stream.read_value<u8>()); | 
					
						
							|  |  |  |     u8 properties = TRY(stream.read_value<u8>()); | 
					
						
							|  |  |  |     header.scale = properties & 0xF; | 
					
						
							|  |  |  |     header.color_encoding = static_cast<ColorEncoding>((properties >> 4) & 0x3); | 
					
						
							|  |  |  |     header.coordinate_range = static_cast<CoordinateRange>((properties >> 6) & 0x3); | 
					
						
							|  |  |  |     switch (header.coordinate_range) { | 
					
						
							|  |  |  |     case CoordinateRange::Default: | 
					
						
							|  |  |  |         header.width = TRY(stream.read_value<LittleEndian<u16>>()); | 
					
						
							|  |  |  |         header.height = TRY(stream.read_value<LittleEndian<u16>>()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case CoordinateRange::Reduced: | 
					
						
							|  |  |  |         header.width = TRY(stream.read_value<u8>()); | 
					
						
							|  |  |  |         header.height = TRY(stream.read_value<u8>()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case CoordinateRange::Enhanced: | 
					
						
							|  |  |  |         header.width = TRY(stream.read_value<LittleEndian<u32>>()); | 
					
						
							|  |  |  |         header.height = TRY(stream.read_value<LittleEndian<u32>>()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return Error::from_string_literal("Invalid TinyVG: Bad coordinate range"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     header.color_count = TRY(stream.read_value<VarUInt>()); | 
					
						
							|  |  |  |     return header; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-31 18:16:27 +00:00
										 |  |  | static ErrorOr<Vector<Color>> decode_color_table(Stream& stream, ColorEncoding encoding, u32 color_count) | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (encoding == ColorEncoding::Custom) | 
					
						
							|  |  |  |         return Error::from_string_literal("Invalid TinyVG: Unsupported color encoding"); | 
					
						
							| 
									
										
										
										
											2023-10-31 18:16:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static constexpr size_t MAX_INITIAL_COLOR_TABLE_SIZE = 65536; | 
					
						
							|  |  |  |     Vector<Color> color_table; | 
					
						
							|  |  |  |     TRY(color_table.try_ensure_capacity(min(MAX_INITIAL_COLOR_TABLE_SIZE, color_count))); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |     auto parse_color = [&]() -> ErrorOr<Color> { | 
					
						
							|  |  |  |         switch (encoding) { | 
					
						
							|  |  |  |         case ColorEncoding::RGBA8888: { | 
					
						
							|  |  |  |             Array<u8, 4> rgba; | 
					
						
							|  |  |  |             TRY(stream.read_until_filled(rgba)); | 
					
						
							|  |  |  |             return Color(rgba[0], rgba[1], rgba[2], rgba[3]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case ColorEncoding::RGB565: { | 
					
						
							|  |  |  |             u16 color = TRY(stream.read_value<LittleEndian<u16>>()); | 
					
						
							|  |  |  |             auto red = (color >> (6 + 5)) & 0x1f; | 
					
						
							|  |  |  |             auto green = (color >> 5) & 0x3f; | 
					
						
							|  |  |  |             auto blue = (color >> 0) & 0x1f; | 
					
						
							| 
									
										
										
										
											2024-03-12 19:31:17 +00:00
										 |  |  |             return Color((red * 255 + 15) / 31, (green * 255 + 31) / 63, (blue * 255 + 15) / 31); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         case ColorEncoding::RGBAF32: { | 
					
						
							|  |  |  |             auto red = TRY(stream.read_value<LittleEndian<f32>>()); | 
					
						
							|  |  |  |             auto green = TRY(stream.read_value<LittleEndian<f32>>()); | 
					
						
							|  |  |  |             auto blue = TRY(stream.read_value<LittleEndian<f32>>()); | 
					
						
							|  |  |  |             auto alpha = TRY(stream.read_value<LittleEndian<f32>>()); | 
					
						
							| 
									
										
										
										
											2023-11-01 17:42:42 +00:00
										 |  |  |             return Color( | 
					
						
							|  |  |  |                 clamp(red * 255.0f, 0.0f, 255.0f), | 
					
						
							|  |  |  |                 clamp(green * 255.0f, 0.0f, 255.0f), | 
					
						
							|  |  |  |                 clamp(blue * 255.0f, 0.0f, 255.0f), | 
					
						
							|  |  |  |                 clamp(alpha * 255.0f, 0.0f, 255.0f)); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return Error::from_string_literal("Invalid TinyVG: Bad color encoding"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2023-10-31 18:16:27 +00:00
										 |  |  |     while (color_count-- > 0) { | 
					
						
							|  |  |  |         TRY(color_table.try_append(TRY(parse_color()))); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return color_table; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TinyVGReader { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     TinyVGReader(Stream& stream, TinyVGHeader const& header, ReadonlySpan<Color> color_table) | 
					
						
							|  |  |  |         : m_stream(stream) | 
					
						
							|  |  |  |         , m_scale(powf(0.5, header.scale)) | 
					
						
							|  |  |  |         , m_coordinate_range(header.coordinate_range) | 
					
						
							|  |  |  |         , m_color_table(color_table) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<float> read_unit() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         auto read_value = [&]() -> ErrorOr<i32> { | 
					
						
							|  |  |  |             switch (m_coordinate_range) { | 
					
						
							|  |  |  |             case CoordinateRange::Default: | 
					
						
							|  |  |  |                 return TRY(m_stream.read_value<LittleEndian<i16>>()); | 
					
						
							|  |  |  |             case CoordinateRange::Reduced: | 
					
						
							|  |  |  |                 return TRY(m_stream.read_value<i8>()); | 
					
						
							|  |  |  |             case CoordinateRange::Enhanced: | 
					
						
							|  |  |  |                 return TRY(m_stream.read_value<LittleEndian<i32>>()); | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 // Note: Already checked while reading the header.
 | 
					
						
							|  |  |  |                 VERIFY_NOT_REACHED(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |         return TRY(read_value()) * m_scale; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<u32> read_var_uint() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return TRY(m_stream.read_value<VarUInt>()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<FloatPoint> read_point() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return FloatPoint { TRY(read_unit()), TRY(read_unit()) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<TinyVGDecodedImageData::Style> read_style(StyleType type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         auto read_color = [&]() -> ErrorOr<Color> { | 
					
						
							|  |  |  |             auto color_index = TRY(m_stream.read_value<VarUInt>()); | 
					
						
							| 
									
										
										
										
											2023-10-02 19:40:23 +01:00
										 |  |  |             if (color_index >= m_color_table.size()) | 
					
						
							|  |  |  |                 return Error::from_string_literal("Invalid color table index"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |             return m_color_table[color_index]; | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2023-07-14 21:34:17 +01:00
										 |  |  |         auto read_gradient = [&]() -> ErrorOr<NonnullRefPtr<SVGGradientPaintStyle>> { | 
					
						
							|  |  |  |             auto point_0 = TRY(read_point()); | 
					
						
							|  |  |  |             auto point_1 = TRY(read_point()); | 
					
						
							|  |  |  |             auto color_0 = TRY(read_color()); | 
					
						
							|  |  |  |             auto color_1 = TRY(read_color()); | 
					
						
							|  |  |  |             // Map TinyVG gradients to SVG gradients (since we already have those).
 | 
					
						
							|  |  |  |             // This is not entirely consistent with the spec, which uses gamma sRGB for gradients
 | 
					
						
							|  |  |  |             // (but this matches the TVG -> SVG renderings).
 | 
					
						
							|  |  |  |             auto svg_gradient = TRY([&]() -> ErrorOr<NonnullRefPtr<SVGGradientPaintStyle>> { | 
					
						
							|  |  |  |                 if (type == StyleType::LinearGradient) | 
					
						
							|  |  |  |                     return TRY(SVGLinearGradientPaintStyle::create(point_0, point_1)); | 
					
						
							|  |  |  |                 auto radius = point_1.distance_from(point_0); | 
					
						
							|  |  |  |                 return TRY(SVGRadialGradientPaintStyle::create(point_0, 0, point_0, radius)); | 
					
						
							|  |  |  |             }()); | 
					
						
							|  |  |  |             TRY(svg_gradient->add_color_stop(0, color_0)); | 
					
						
							|  |  |  |             TRY(svg_gradient->add_color_stop(1, color_1)); | 
					
						
							|  |  |  |             return svg_gradient; | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |         switch (type) { | 
					
						
							| 
									
										
										
										
											2023-07-14 21:34:17 +01:00
										 |  |  |         case StyleType::FlatColored: | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |             return TRY(read_color()); | 
					
						
							|  |  |  |         case StyleType::LinearGradient: | 
					
						
							| 
									
										
										
										
											2023-07-14 21:34:17 +01:00
										 |  |  |         case StyleType::RadialGradinet: | 
					
						
							|  |  |  |             return TRY(read_gradient()); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return Error::from_string_literal("Invalid TinyVG: Bad style data"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<FloatRect> read_rectangle() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return FloatRect { TRY(read_unit()), TRY(read_unit()), TRY(read_unit()), TRY(read_unit()) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<FloatLine> read_line() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return FloatLine { TRY(read_point()), TRY(read_point()) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<Path> read_path(u32 segment_count) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Path path; | 
					
						
							|  |  |  |         auto segment_lengths = TRY(FixedArray<u32>::create(segment_count)); | 
					
						
							|  |  |  |         for (auto& command_count : segment_lengths) { | 
					
						
							|  |  |  |             command_count = TRY(read_var_uint()) + 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         for (auto command_count : segment_lengths) { | 
					
						
							|  |  |  |             auto start_point = TRY(read_point()); | 
					
						
							|  |  |  |             path.move_to(start_point); | 
					
						
							|  |  |  |             for (u32 i = 0; i < command_count; i++) { | 
					
						
							|  |  |  |                 u8 command_tag = TRY(m_stream.read_value<u8>()); | 
					
						
							|  |  |  |                 auto path_command = static_cast<PathCommand>(command_tag & 0x7); | 
					
						
							| 
									
										
										
										
											2023-07-14 21:31:42 +01:00
										 |  |  |                 bool has_line_width = (command_tag >> 4) & 0b1; | 
					
						
							|  |  |  |                 if (has_line_width) { | 
					
						
							|  |  |  |                     // FIXME: TinyVG allows changing the line width within a path.
 | 
					
						
							|  |  |  |                     // This is not supported in LibGfx, so we currently ignore this.
 | 
					
						
							|  |  |  |                     (void)TRY(read_unit()); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 switch (path_command) { | 
					
						
							|  |  |  |                 case PathCommand::Line: | 
					
						
							|  |  |  |                     path.line_to(TRY(read_point())); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case PathCommand::HorizontalLine: | 
					
						
							| 
									
										
										
										
											2024-03-17 20:23:17 +00:00
										 |  |  |                     path.line_to({ TRY(read_unit()), path.last_point().y() }); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 case PathCommand::VerticalLine: | 
					
						
							| 
									
										
										
										
											2024-03-17 20:23:17 +00:00
										 |  |  |                     path.line_to({ path.last_point().x(), TRY(read_unit()) }); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 case PathCommand::CubicBezier: { | 
					
						
							|  |  |  |                     auto control_0 = TRY(read_point()); | 
					
						
							|  |  |  |                     auto control_1 = TRY(read_point()); | 
					
						
							|  |  |  |                     auto point_1 = TRY(read_point()); | 
					
						
							|  |  |  |                     path.cubic_bezier_curve_to(control_0, control_1, point_1); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 case PathCommand::ArcCircle: { | 
					
						
							|  |  |  |                     u8 flags = TRY(m_stream.read_value<u8>()); | 
					
						
							|  |  |  |                     bool large_arc = (flags >> 0) & 0b1; | 
					
						
							|  |  |  |                     bool sweep = (flags >> 1) & 0b1; | 
					
						
							|  |  |  |                     auto radius = TRY(read_unit()); | 
					
						
							|  |  |  |                     auto target = TRY(read_point()); | 
					
						
							|  |  |  |                     path.arc_to(target, radius, large_arc, !sweep); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 case PathCommand::ArcEllipse: { | 
					
						
							|  |  |  |                     u8 flags = TRY(m_stream.read_value<u8>()); | 
					
						
							|  |  |  |                     bool large_arc = (flags >> 0) & 0b1; | 
					
						
							|  |  |  |                     bool sweep = (flags >> 1) & 0b1; | 
					
						
							|  |  |  |                     auto radius_x = TRY(read_unit()); | 
					
						
							|  |  |  |                     auto radius_y = TRY(read_unit()); | 
					
						
							|  |  |  |                     auto rotation = TRY(read_unit()); | 
					
						
							|  |  |  |                     auto target = TRY(read_point()); | 
					
						
							|  |  |  |                     path.elliptical_arc_to(target, { radius_x, radius_y }, rotation, large_arc, !sweep); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 case PathCommand::ClosePath: { | 
					
						
							|  |  |  |                     path.close(); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 case PathCommand::QuadraticBezier: { | 
					
						
							|  |  |  |                     auto control = TRY(read_point()); | 
					
						
							|  |  |  |                     auto point_1 = TRY(read_point()); | 
					
						
							|  |  |  |                     path.quadratic_bezier_curve_to(control, point_1); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return Error::from_string_literal("Invalid TinyVG: Bad path command"); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return path; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<FillCommandHeader> read_fill_command_header(StyleType style_type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return FillCommandHeader { TRY(read_var_uint()) + 1, TRY(read_style(style_type)) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<DrawCommandHeader> read_draw_command_header(StyleType style_type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return DrawCommandHeader { TRY(read_var_uint()) + 1, TRY(read_style(style_type)), TRY(read_unit()) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ErrorOr<OutlineFillCommandHeader> read_outline_fill_command_header(StyleType style_type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         u8 header = TRY(m_stream.read_value<u8>()); | 
					
						
							|  |  |  |         u8 count = (header & 0x3f) + 1; | 
					
						
							|  |  |  |         auto stroke_type = static_cast<StyleType>((header >> 6) & 0x3); | 
					
						
							|  |  |  |         return OutlineFillCommandHeader { count, TRY(read_style(style_type)), TRY(read_style(stroke_type)), TRY(read_unit()) }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     Stream& m_stream; | 
					
						
							|  |  |  |     float m_scale {}; | 
					
						
							|  |  |  |     CoordinateRange m_coordinate_range; | 
					
						
							|  |  |  |     ReadonlySpan<Color> m_color_table; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  | ErrorOr<NonnullRefPtr<TinyVGDecodedImageData>> TinyVGDecodedImageData::decode(Stream& stream) | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     return decode(stream, TRY(decode_tinyvg_header(stream))); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ErrorOr<NonnullRefPtr<TinyVGDecodedImageData>> TinyVGDecodedImageData::decode(Stream& stream, TinyVGHeader const& header) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |     if (header.version != 1) | 
					
						
							|  |  |  |         return Error::from_string_literal("Invalid TinyVG: Unsupported version"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-12 19:26:41 +00:00
										 |  |  |     auto color_table = TRY(decode_color_table(stream, header.color_encoding, header.color_count)); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |     TinyVGReader reader { stream, header, color_table.span() }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto rectangle_to_path = [](FloatRect const& rect) -> Path { | 
					
						
							|  |  |  |         Path path; | 
					
						
							|  |  |  |         path.move_to({ rect.x(), rect.y() }); | 
					
						
							|  |  |  |         path.line_to({ rect.x() + rect.width(), rect.y() }); | 
					
						
							|  |  |  |         path.line_to({ rect.x() + rect.width(), rect.y() + rect.height() }); | 
					
						
							|  |  |  |         path.line_to({ rect.x(), rect.y() + rect.height() }); | 
					
						
							|  |  |  |         path.close(); | 
					
						
							|  |  |  |         return path; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Vector<DrawCommand> draw_commands; | 
					
						
							|  |  |  |     bool at_end = false; | 
					
						
							|  |  |  |     while (!at_end) { | 
					
						
							|  |  |  |         u8 command_info = TRY(stream.read_value<u8>()); | 
					
						
							|  |  |  |         auto command = static_cast<Command>(command_info & 0x3f); | 
					
						
							|  |  |  |         auto style_type = static_cast<StyleType>((command_info >> 6) & 0x3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (command) { | 
					
						
							|  |  |  |         case Command::EndOfDocument: | 
					
						
							|  |  |  |             at_end = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Command::FillPolygon: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_fill_command_header(style_type)); | 
					
						
							|  |  |  |             Path polygon; | 
					
						
							|  |  |  |             polygon.move_to(TRY(reader.read_point())); | 
					
						
							|  |  |  |             for (u32 i = 0; i < header.count - 1; i++) | 
					
						
							|  |  |  |                 polygon.line_to(TRY(reader.read_point())); | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(polygon), move(header.style) })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::FillRectangles: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_fill_command_header(style_type)); | 
					
						
							|  |  |  |             for (u32 i = 0; i < header.count; i++) { | 
					
						
							|  |  |  |                 TRY(draw_commands.try_append(DrawCommand { | 
					
						
							| 
									
										
										
										
											2023-07-14 21:33:22 +01:00
										 |  |  |                     rectangle_to_path(TRY(reader.read_rectangle())), header.style })); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::FillPath: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_fill_command_header(style_type)); | 
					
						
							|  |  |  |             auto path = TRY(reader.read_path(header.count)); | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(path), move(header.style) })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::DrawLines: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_draw_command_header(style_type)); | 
					
						
							|  |  |  |             Path path; | 
					
						
							|  |  |  |             for (u32 i = 0; i < header.count; i++) { | 
					
						
							|  |  |  |                 auto line = TRY(reader.read_line()); | 
					
						
							|  |  |  |                 path.move_to(line.a()); | 
					
						
							|  |  |  |                 path.line_to(line.b()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(path), {}, move(header.line_style), header.line_width })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::DrawLineStrip: | 
					
						
							|  |  |  |         case Command::DrawLineLoop: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_draw_command_header(style_type)); | 
					
						
							|  |  |  |             Path path; | 
					
						
							|  |  |  |             path.move_to(TRY(reader.read_point())); | 
					
						
							|  |  |  |             for (u32 i = 0; i < header.count - 1; i++) | 
					
						
							|  |  |  |                 path.line_to(TRY(reader.read_point())); | 
					
						
							|  |  |  |             if (command == Command::DrawLineLoop) | 
					
						
							|  |  |  |                 path.close(); | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(path), {}, move(header.line_style), header.line_width })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::DrawLinePath: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_draw_command_header(style_type)); | 
					
						
							|  |  |  |             auto path = TRY(reader.read_path(header.count)); | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(path), {}, move(header.line_style), header.line_width })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::OutlineFillPolygon: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_outline_fill_command_header(style_type)); | 
					
						
							|  |  |  |             Path polygon; | 
					
						
							|  |  |  |             polygon.move_to(TRY(reader.read_point())); | 
					
						
							|  |  |  |             for (u32 i = 0; i < header.count - 1; i++) | 
					
						
							|  |  |  |                 polygon.line_to(TRY(reader.read_point())); | 
					
						
							| 
									
										
										
										
											2023-07-14 22:55:57 +01:00
										 |  |  |             polygon.close(); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |             TRY(draw_commands.try_append(DrawCommand { move(polygon), move(header.fill_style), move(header.line_style), header.line_width })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::OutlineFillRectangles: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_outline_fill_command_header(style_type)); | 
					
						
							| 
									
										
										
										
											2023-07-14 21:33:22 +01:00
										 |  |  |             for (u32 i = 0; i < header.count; i++) { | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 TRY(draw_commands.try_append(DrawCommand { | 
					
						
							| 
									
										
										
										
											2023-07-14 21:33:22 +01:00
										 |  |  |                     rectangle_to_path(TRY(reader.read_rectangle())), header.fill_style, header.line_style, header.line_width })); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case Command::OutLineFillPath: { | 
					
						
							|  |  |  |             auto header = TRY(reader.read_outline_fill_command_header(style_type)); | 
					
						
							|  |  |  |             auto path = TRY(reader.read_path(header.count)); | 
					
						
							|  |  |  |             TRY(draw_commands.try_append(DrawCommand { move(path), move(header.fill_style), move(header.line_style), header.line_width })); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return Error::from_string_literal("Invalid TinyVG: Bad command"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  |     return TRY(adopt_nonnull_ref_or_enomem(new (nothrow) TinyVGDecodedImageData({ header.width, header.height }, move(draw_commands)))); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  | void TinyVGDecodedImageData::draw_transformed(Painter& painter, AffineTransform transform) const | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  |     // FIXME: Correctly handle non-uniform scales.
 | 
					
						
							|  |  |  |     auto scale = max(transform.x_scale(), transform.y_scale()); | 
					
						
							|  |  |  |     AntiAliasingPainter aa_painter { painter }; | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |     for (auto const& command : draw_commands()) { | 
					
						
							|  |  |  |         auto draw_path = command.path.copy_transformed(transform); | 
					
						
							|  |  |  |         if (command.fill.has_value()) { | 
					
						
							|  |  |  |             auto fill_path = draw_path; | 
					
						
							|  |  |  |             fill_path.close_all_subpaths(); | 
					
						
							|  |  |  |             command.fill->visit( | 
					
						
							| 
									
										
										
										
											2024-06-05 10:21:28 +02:00
										 |  |  |                 [&](Color color) { aa_painter.fill_path(fill_path, color, WindingRule::EvenOdd); }, | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 [&](NonnullRefPtr<SVGGradientPaintStyle> style) { | 
					
						
							|  |  |  |                     const_cast<SVGGradientPaintStyle&>(*style).set_gradient_transform(transform); | 
					
						
							| 
									
										
										
										
											2024-06-05 10:21:28 +02:00
										 |  |  |                     aa_painter.fill_path(fill_path, style, 1.0f, WindingRule::EvenOdd); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (command.stroke.has_value()) { | 
					
						
							|  |  |  |             command.stroke->visit( | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  |                 [&](Color color) { aa_painter.stroke_path(draw_path, color, command.stroke_width * scale); }, | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 [&](NonnullRefPtr<SVGGradientPaintStyle> style) { | 
					
						
							|  |  |  |                     const_cast<SVGGradientPaintStyle&>(*style).set_gradient_transform(transform); | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  |                     aa_painter.stroke_path(draw_path, style, command.stroke_width * scale); | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  |                 }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  | struct TinyVGLoadingContext { | 
					
						
							|  |  |  |     FixedMemoryStream stream; | 
					
						
							|  |  |  |     TinyVGHeader header {}; | 
					
						
							|  |  |  |     RefPtr<TinyVGDecodedImageData> decoded_image {}; | 
					
						
							|  |  |  |     RefPtr<Bitmap> bitmap {}; | 
					
						
							|  |  |  |     enum class State { | 
					
						
							|  |  |  |         NotDecoded = 0, | 
					
						
							|  |  |  |         HeaderDecoded, | 
					
						
							|  |  |  |         ImageDecoded, | 
					
						
							|  |  |  |         Error, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     State state { State::NotDecoded }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ErrorOr<void> decode_header_and_update_context(TinyVGLoadingContext& context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     VERIFY(context.state == TinyVGLoadingContext::State::NotDecoded); | 
					
						
							| 
									
										
										
										
											2023-07-14 12:30:45 -04:00
										 |  |  |     context.header = TRY(decode_tinyvg_header(context.stream)); | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     context.state = TinyVGLoadingContext::State::HeaderDecoded; | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ErrorOr<void> decode_image_data_and_update_context(TinyVGLoadingContext& context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     VERIFY(context.state == TinyVGLoadingContext::State::HeaderDecoded); | 
					
						
							|  |  |  |     auto image_data_or_error = TinyVGDecodedImageData::decode(context.stream, context.header); | 
					
						
							|  |  |  |     if (image_data_or_error.is_error()) { | 
					
						
							|  |  |  |         context.state = TinyVGLoadingContext::State::Error; | 
					
						
							|  |  |  |         return image_data_or_error.release_error(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     context.state = TinyVGLoadingContext::State::ImageDecoded; | 
					
						
							|  |  |  |     context.decoded_image = image_data_or_error.release_value(); | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ErrorOr<void> ensure_fully_decoded(TinyVGLoadingContext& context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (context.state == TinyVGLoadingContext::State::Error) | 
					
						
							|  |  |  |         return Error::from_string_literal("TinyVGImageDecoderPlugin: Decoding failed!"); | 
					
						
							|  |  |  |     if (context.state == TinyVGLoadingContext::State::HeaderDecoded) | 
					
						
							|  |  |  |         TRY(decode_image_data_and_update_context(context)); | 
					
						
							|  |  |  |     VERIFY(context.state == TinyVGLoadingContext::State::ImageDecoded); | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | TinyVGImageDecoderPlugin::TinyVGImageDecoderPlugin(ReadonlyBytes bytes) | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     : m_context { make<TinyVGLoadingContext>(FixedMemoryStream { bytes }) } | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> TinyVGImageDecoderPlugin::create(ReadonlyBytes bytes) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-14 12:30:45 -04:00
										 |  |  |     auto plugin = TRY(adopt_nonnull_own_or_enomem(new (nothrow) TinyVGImageDecoderPlugin(bytes))); | 
					
						
							|  |  |  |     TRY(decode_header_and_update_context(*plugin->m_context)); | 
					
						
							|  |  |  |     return plugin; | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TinyVGImageDecoderPlugin::sniff(ReadonlyBytes bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FixedMemoryStream stream { { bytes.data(), bytes.size() } }; | 
					
						
							|  |  |  |     return !decode_tinyvg_header(stream).is_error(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IntSize TinyVGImageDecoderPlugin::size() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     return { m_context->header.width, m_context->header.height }; | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ErrorOr<ImageFrameDescriptor> TinyVGImageDecoderPlugin::frame(size_t, Optional<IntSize> ideal_size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     TRY(ensure_fully_decoded(*m_context)); | 
					
						
							|  |  |  |     auto target_size = ideal_size.value_or(m_context->decoded_image->size()); | 
					
						
							|  |  |  |     if (!m_context->bitmap || m_context->bitmap->size() != target_size) | 
					
						
							|  |  |  |         m_context->bitmap = TRY(m_context->decoded_image->bitmap(target_size)); | 
					
						
							|  |  |  |     return ImageFrameDescriptor { m_context->bitmap }; | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  | ErrorOr<VectorImageFrameDescriptor> TinyVGImageDecoderPlugin::vector_frame(size_t) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-08 03:29:51 +01:00
										 |  |  |     TRY(ensure_fully_decoded(*m_context)); | 
					
						
							|  |  |  |     return VectorImageFrameDescriptor { m_context->decoded_image, 0 }; | 
					
						
							| 
									
										
										
										
											2023-07-07 22:23:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-02 22:24:01 +01:00
										 |  |  | } |