mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 01:41:59 -05:00
d569238896
This adds support for dash patterns to Path::stroke_to_fill(). This is used in PDFs, <canvas>, and <svg>. The implementation is based on the <canvas> spec. It seems to do the right thing for PDF files too. (This commit only adds the feature to LibGfx. Future commits will hook this up for PDF, <canvas>, and <svg>.)
157 lines
4.9 KiB
C++
157 lines
4.9 KiB
C++
/*
|
|
* Copyright (c) 2024, Nico Weber <thakis@chromium.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <LibTest/TestCase.h>
|
|
|
|
#include <LibGfx/Path.h>
|
|
|
|
TEST_CASE(path_to_fill_short_wide_line_with_butt_linecap)
|
|
{
|
|
// Test drawing a horizontal rect by stroking a vertical short wide line.
|
|
{
|
|
float width = 100;
|
|
float height = 1;
|
|
Gfx::Path path;
|
|
path.move_to({ width / 2, 0 });
|
|
path.line_to({ width / 2, height });
|
|
auto fill = path.stroke_to_fill({ width, Gfx::Path::CapStyle::Butt });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 0, 0 }, { width, height }));
|
|
}
|
|
|
|
// Test drawing a vertical rect by stroking a horizontal short wide line.
|
|
{
|
|
float width = 1;
|
|
float height = 100;
|
|
Gfx::Path path;
|
|
path.move_to({ 0, height / 2 });
|
|
path.line_to({ width, height / 2 });
|
|
auto fill = path.stroke_to_fill({ height, Gfx::Path::CapStyle::Butt });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 0, 0 }, { width, height }));
|
|
}
|
|
}
|
|
TEST_CASE(path_to_fill_square_linecap)
|
|
{
|
|
float line_width = 10;
|
|
float width = 100;
|
|
Gfx::Path path;
|
|
path.move_to({ line_width / 2, line_width / 2 });
|
|
path.line_to({ width - line_width / 2, line_width / 2 });
|
|
auto fill = path.stroke_to_fill({ line_width, Gfx::Path::CapStyle::Square });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 0, 0 }, { width, line_width }));
|
|
}
|
|
|
|
TEST_CASE(path_to_fill_single_point)
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 10, 10 });
|
|
path.line_to({ 10, 10 });
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Butt });
|
|
EXPECT(fill.is_empty());
|
|
}
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Round });
|
|
EXPECT(!fill.is_empty());
|
|
}
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Square });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 6, 6 }, { 8, 8 }));
|
|
}
|
|
}
|
|
|
|
TEST_CASE(path_to_fill_two_single_points)
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 10, 10 });
|
|
path.line_to({ 10, 10 });
|
|
path.move_to({ 20, 20 });
|
|
path.line_to({ 20, 20 });
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Butt });
|
|
EXPECT(fill.is_empty());
|
|
}
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Round });
|
|
EXPECT(!fill.is_empty());
|
|
}
|
|
|
|
{
|
|
auto fill = path.stroke_to_fill({ 8, Gfx::Path::CapStyle::Square });
|
|
EXPECT(!fill.is_empty());
|
|
}
|
|
}
|
|
|
|
TEST_CASE(path_to_fill_miter_linejoin)
|
|
{
|
|
float line_width = 2;
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 0, 0 });
|
|
path.line_to({ 2, 0 });
|
|
path.line_to({ 2, 2 });
|
|
auto fill = path.stroke_to_fill({ line_width, Gfx::Path::CapStyle::Butt, Gfx::Path::JoinStyle::Miter });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 0, -1 }, { 3, 3 }));
|
|
EXPECT_EQ(fill.to_byte_string(), "M 1,1 L 1,2 L 3,2 L 3,-1 L 0,-1 L 0,1 L 1,1 Z");
|
|
}
|
|
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 1, 1 });
|
|
path.line_to({ 4, 1 });
|
|
path.line_to({ 4, 4 });
|
|
path.line_to({ 1, 4 });
|
|
path.close();
|
|
auto fill = path.stroke_to_fill({ line_width, Gfx::Path::CapStyle::Butt, Gfx::Path::JoinStyle::Miter });
|
|
EXPECT_EQ(fill.bounding_box(), Gfx::FloatRect({ 0, 0 }, { 5, 5 }));
|
|
EXPECT_EQ(fill.to_byte_string(), "M 3,2 L 3,3 L 2,3 L 2,2 L 3,2 Z M 0,5 L 5,5 L 5,0 L 0,0 L 0,5 Z");
|
|
}
|
|
}
|
|
|
|
TEST_CASE(path_to_fill_dash)
|
|
{
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 0, 0.5 });
|
|
path.line_to({ 13, 0.5 });
|
|
auto fill = path.stroke_to_fill({ .thickness = 1, .cap_style = Gfx::Path::CapStyle::Butt, .dash_pattern = { 3, 3 }, .dash_offset = 0 });
|
|
EXPECT_EQ(fill.to_byte_string(), "M 3,1 L 3,0 L 0,0 L 0,1 L 3,1 Z M 9,1 L 9,0 L 6,0 L 6,1 L 9,1 Z M 13,1 L 13,0 L 12,0 L 12,1 L 13,1 Z");
|
|
}
|
|
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 0, 0.5 });
|
|
path.line_to({ 13, 0.5 });
|
|
auto fill = path.stroke_to_fill({ .thickness = 1, .cap_style = Gfx::Path::CapStyle::Butt, .dash_pattern = { 2, 3 }, .dash_offset = 11 });
|
|
EXPECT_EQ(fill.to_byte_string(), "M 1,1 L 1,0 L 0,0 L 0,1 L 1,1 Z M 6,1 L 6,0 L 4,0 L 4,1 L 6,1 Z M 11,1 L 11,0 L 9,0 L 9,1 L 11,1 Z");
|
|
}
|
|
}
|
|
|
|
TEST_CASE(path_to_string)
|
|
{
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 10, 10 });
|
|
path.line_to({ 20, 20 });
|
|
path.quadratic_bezier_curve_to({ 30, 30 }, { 40, 40 });
|
|
path.cubic_bezier_curve_to({ 50, 50 }, { 60, 60 }, { 10, 10 });
|
|
path.close();
|
|
EXPECT_EQ(path.to_byte_string(), "M 10,10 L 20,20 Q 30,30 40,40 C 50,50 60,60 10,10 Z");
|
|
}
|
|
|
|
{
|
|
Gfx::Path path;
|
|
path.move_to({ 10, 10 });
|
|
path.line_to({ 20, 20 });
|
|
path.quadratic_bezier_curve_to({ 30, 30 }, { 40, 40 });
|
|
path.cubic_bezier_curve_to({ 50, 50 }, { 60, 60 }, { 10, 10 });
|
|
EXPECT_EQ(path.to_byte_string(), "M 10,10 L 20,20 Q 30,30 40,40 C 50,50 60,60 10,10");
|
|
}
|
|
}
|