mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-23 01:32:14 -05:00
211 lines
5 KiB
C++
211 lines
5 KiB
C++
/*
|
|
* Copyright (c) 2020, Matthew Olsson <mattco@serenityos.org>
|
|
* Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
|
|
* Copyright (c) 2024, Tim Ledbetter <timledbetter@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/GenericLexer.h>
|
|
#include <AK/Variant.h>
|
|
#include <AK/Vector.h>
|
|
#include <LibGfx/Point.h>
|
|
|
|
namespace Web::SVG {
|
|
|
|
enum class PathInstructionType {
|
|
Move,
|
|
ClosePath,
|
|
Line,
|
|
HorizontalLine,
|
|
VerticalLine,
|
|
Curve,
|
|
SmoothCurve,
|
|
QuadraticBezierCurve,
|
|
SmoothQuadraticBezierCurve,
|
|
EllipticalArc,
|
|
Invalid,
|
|
};
|
|
|
|
struct PathInstruction {
|
|
PathInstructionType type;
|
|
bool absolute;
|
|
Vector<float> data;
|
|
};
|
|
|
|
struct Transform {
|
|
struct Translate {
|
|
float x;
|
|
float y;
|
|
};
|
|
struct Scale {
|
|
float x;
|
|
float y;
|
|
};
|
|
struct Rotate {
|
|
float a;
|
|
float x;
|
|
float y;
|
|
};
|
|
struct SkewX {
|
|
float a;
|
|
};
|
|
struct SkewY {
|
|
float a;
|
|
};
|
|
struct Matrix {
|
|
float a;
|
|
float b;
|
|
float c;
|
|
float d;
|
|
float e;
|
|
float f;
|
|
};
|
|
|
|
using Operation = Variant<Translate, Scale, Rotate, SkewX, SkewY, Matrix>;
|
|
Operation operation;
|
|
};
|
|
|
|
struct PreserveAspectRatio {
|
|
enum class Align {
|
|
None,
|
|
xMinYMin,
|
|
xMidYMin,
|
|
xMaxYMin,
|
|
xMinYMid,
|
|
xMidYMid,
|
|
xMaxYMid,
|
|
xMinYMax,
|
|
xMidYMax,
|
|
xMaxYMax
|
|
};
|
|
enum class MeetOrSlice {
|
|
Meet,
|
|
Slice
|
|
};
|
|
Align align { Align::xMidYMid };
|
|
MeetOrSlice meet_or_slice { MeetOrSlice::Meet };
|
|
};
|
|
|
|
enum class SVGUnits {
|
|
ObjectBoundingBox,
|
|
UserSpaceOnUse
|
|
};
|
|
|
|
using GradientUnits = SVGUnits;
|
|
using MaskUnits = SVGUnits;
|
|
using MaskContentUnits = SVGUnits;
|
|
using ClipPathUnits = SVGUnits;
|
|
|
|
enum class SpreadMethod {
|
|
Pad,
|
|
Repeat,
|
|
Reflect
|
|
};
|
|
|
|
class NumberPercentage {
|
|
public:
|
|
NumberPercentage(float value, bool is_percentage)
|
|
: m_value(is_percentage ? value / 100 : value)
|
|
, m_is_percentage(is_percentage)
|
|
{
|
|
}
|
|
|
|
static NumberPercentage create_percentage(float value)
|
|
{
|
|
return NumberPercentage(value, true);
|
|
}
|
|
|
|
static NumberPercentage create_number(float value)
|
|
{
|
|
return NumberPercentage(value, false);
|
|
}
|
|
|
|
float resolve_relative_to(float length) const;
|
|
|
|
float value() const { return m_value; }
|
|
|
|
private:
|
|
float m_value;
|
|
bool m_is_percentage { false };
|
|
};
|
|
|
|
enum class FillRule {
|
|
Nonzero,
|
|
Evenodd
|
|
};
|
|
|
|
using ClipRule = FillRule;
|
|
|
|
enum class TextAnchor {
|
|
Start,
|
|
Middle,
|
|
End
|
|
};
|
|
|
|
class AttributeParser final {
|
|
public:
|
|
~AttributeParser() = default;
|
|
|
|
static Optional<float> parse_coordinate(StringView input);
|
|
static Optional<float> parse_length(StringView input);
|
|
static Optional<NumberPercentage> parse_number_percentage(StringView input);
|
|
static Optional<float> parse_positive_length(StringView input);
|
|
static Vector<Gfx::FloatPoint> parse_points(StringView input);
|
|
static Vector<PathInstruction> parse_path_data(StringView input);
|
|
static Optional<Vector<Transform>> parse_transform(StringView input);
|
|
static Optional<PreserveAspectRatio> parse_preserve_aspect_ratio(StringView input);
|
|
static Optional<SVGUnits> parse_units(StringView input);
|
|
static Optional<SpreadMethod> parse_spread_method(StringView input);
|
|
|
|
private:
|
|
AttributeParser(StringView source);
|
|
|
|
ErrorOr<void> parse_drawto();
|
|
ErrorOr<void> parse_moveto();
|
|
void parse_closepath();
|
|
ErrorOr<void> parse_lineto();
|
|
ErrorOr<void> parse_horizontal_lineto();
|
|
ErrorOr<void> parse_vertical_lineto();
|
|
ErrorOr<void> parse_curveto();
|
|
ErrorOr<void> parse_smooth_curveto();
|
|
ErrorOr<void> parse_quadratic_bezier_curveto();
|
|
ErrorOr<void> parse_smooth_quadratic_bezier_curveto();
|
|
ErrorOr<void> parse_elliptical_arc();
|
|
|
|
Optional<Vector<Transform>> parse_transform();
|
|
|
|
ErrorOr<float> parse_length();
|
|
ErrorOr<float> parse_coordinate();
|
|
ErrorOr<Vector<float>> parse_coordinate_pair();
|
|
ErrorOr<Vector<float>> parse_coordinate_sequence();
|
|
ErrorOr<Vector<Vector<float>>> parse_coordinate_pair_sequence();
|
|
ErrorOr<Vector<float>> parse_coordinate_pair_double();
|
|
ErrorOr<Vector<float>> parse_coordinate_pair_triplet();
|
|
ErrorOr<Vector<float>> parse_elliptical_arc_argument();
|
|
void parse_whitespace(bool must_match_once = false);
|
|
void parse_comma_whitespace();
|
|
ErrorOr<float> parse_number();
|
|
ErrorOr<float> parse_nonnegative_number();
|
|
ErrorOr<float> parse_flag();
|
|
// -1 if negative, +1 otherwise
|
|
int parse_sign();
|
|
|
|
bool match_whitespace() const;
|
|
bool match_comma_whitespace() const;
|
|
bool match_coordinate() const;
|
|
bool match_length() const;
|
|
bool match_number() const;
|
|
bool match(char c) const { return !done() && ch() == c; }
|
|
|
|
bool done() const { return m_lexer.is_eof(); }
|
|
char ch(size_t offset = 0) const { return m_lexer.peek(offset); }
|
|
char consume() { return m_lexer.consume(); }
|
|
|
|
GenericLexer m_lexer;
|
|
Vector<PathInstruction> m_instructions;
|
|
};
|
|
|
|
}
|