mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-07 19:57:45 +03:00
538 lines
20 KiB
C++
538 lines
20 KiB
C++
/*
|
|
* Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org>
|
|
* Copyright (c) 2022, Ben Maxwell <macdue@dueutil.tech>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#if defined(__GNUC__) && !defined(__clang__)
|
|
# pragma GCC optimize("O3")
|
|
#endif
|
|
|
|
#include "FillPathImplementation.h"
|
|
#include <AK/Function.h>
|
|
#include <AK/NumericLimits.h>
|
|
#include <LibGfx/AntiAliasingPainter.h>
|
|
#include <LibGfx/Path.h>
|
|
|
|
// Base algorithm from https://en.wikipedia.org/wiki/Xiaolin_Wu%27s_line_algorithm,
|
|
// because there seems to be no other known method for drawing AA'd lines (?)
|
|
template<Gfx::AntiAliasingPainter::AntiAliasPolicy policy>
|
|
void Gfx::AntiAliasingPainter::draw_anti_aliased_line(FloatPoint const& actual_from, FloatPoint const& actual_to, Color color, float thickness, Gfx::Painter::LineStyle style, Color)
|
|
{
|
|
// FIXME: Implement this :P
|
|
VERIFY(style == Painter::LineStyle::Solid);
|
|
|
|
auto corrected_thickness = thickness > 1 ? thickness - 1 : thickness;
|
|
auto size = IntSize(corrected_thickness, corrected_thickness);
|
|
auto plot = [&](int x, int y, float c) {
|
|
m_underlying_painter.fill_rect(IntRect::centered_on({ x, y }, size), color.with_alpha(color.alpha() * c));
|
|
};
|
|
|
|
auto integer_part = [](float x) { return floorf(x); };
|
|
auto round = [&](float x) { return integer_part(x + 0.5f); };
|
|
auto fractional_part = [&](float x) { return x - floorf(x); };
|
|
auto one_minus_fractional_part = [&](float x) { return 1.0f - fractional_part(x); };
|
|
|
|
auto draw_line = [&](float x0, float y0, float x1, float y1) {
|
|
bool steep = fabsf(y1 - y0) > fabsf(x1 - x0);
|
|
|
|
if (steep) {
|
|
swap(x0, y0);
|
|
swap(x1, y1);
|
|
}
|
|
|
|
if (x0 > x1) {
|
|
swap(x0, x1);
|
|
swap(y0, y1);
|
|
}
|
|
|
|
float dx = x1 - x0;
|
|
float dy = y1 - y0;
|
|
|
|
float gradient;
|
|
if (dx == 0.0f)
|
|
gradient = 1.0f;
|
|
else
|
|
gradient = dy / dx;
|
|
|
|
// Handle first endpoint.
|
|
int x_end = round(x0);
|
|
int y_end = y0 + gradient * (x_end - x0);
|
|
float x_gap = one_minus_fractional_part(x0 + 0.5f);
|
|
|
|
int xpxl1 = x_end; // This will be used in the main loop.
|
|
int ypxl1 = integer_part(y_end);
|
|
|
|
if (steep) {
|
|
plot(ypxl1, xpxl1, one_minus_fractional_part(y_end) * x_gap);
|
|
plot(ypxl1 + 1, xpxl1, fractional_part(y_end) * x_gap);
|
|
} else {
|
|
plot(xpxl1, ypxl1, one_minus_fractional_part(y_end) * x_gap);
|
|
plot(xpxl1, ypxl1 + 1, fractional_part(y_end) * x_gap);
|
|
}
|
|
|
|
float intery = y_end + gradient; // First y-intersection for the main loop.
|
|
|
|
// Handle second endpoint.
|
|
x_end = round(x1);
|
|
y_end = y1 + gradient * (x_end - x1);
|
|
x_gap = fractional_part(x1 + 0.5f);
|
|
int xpxl2 = x_end; // This will be used in the main loop
|
|
int ypxl2 = integer_part(y_end);
|
|
|
|
if (steep) {
|
|
plot(ypxl2, xpxl2, one_minus_fractional_part(y_end) * x_gap);
|
|
plot(ypxl2 + 1, xpxl2, fractional_part(y_end) * x_gap);
|
|
} else {
|
|
plot(xpxl2, ypxl2, one_minus_fractional_part(y_end) * x_gap);
|
|
plot(xpxl2, ypxl2 + 1, fractional_part(y_end) * x_gap);
|
|
}
|
|
|
|
// Main loop.
|
|
if (steep) {
|
|
for (int x = xpxl1 + 1; x <= xpxl2 - 1; ++x) {
|
|
if constexpr (policy == AntiAliasPolicy::OnlyEnds) {
|
|
plot(integer_part(intery), x, 1);
|
|
} else {
|
|
plot(integer_part(intery), x, one_minus_fractional_part(intery));
|
|
}
|
|
plot(integer_part(intery) + 1, x, fractional_part(intery));
|
|
intery += gradient;
|
|
}
|
|
} else {
|
|
for (int x = xpxl1 + 1; x <= xpxl2 - 1; ++x) {
|
|
if constexpr (policy == AntiAliasPolicy::OnlyEnds) {
|
|
plot(x, integer_part(intery), 1);
|
|
} else {
|
|
plot(x, integer_part(intery), one_minus_fractional_part(intery));
|
|
}
|
|
plot(x, integer_part(intery) + 1, fractional_part(intery));
|
|
intery += gradient;
|
|
}
|
|
}
|
|
};
|
|
|
|
auto mapped_from = m_transform.map(actual_from);
|
|
auto mapped_to = m_transform.map(actual_to);
|
|
draw_line(mapped_from.x(), mapped_from.y(), mapped_to.x(), mapped_to.y());
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_aliased_line(FloatPoint const& actual_from, FloatPoint const& actual_to, Color color, float thickness, Gfx::Painter::LineStyle style, Color alternate_color)
|
|
{
|
|
draw_anti_aliased_line<AntiAliasPolicy::OnlyEnds>(actual_from, actual_to, color, thickness, style, alternate_color);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_line(FloatPoint const& actual_from, FloatPoint const& actual_to, Color color, float thickness, Gfx::Painter::LineStyle style, Color alternate_color)
|
|
{
|
|
draw_anti_aliased_line<AntiAliasPolicy::Full>(actual_from, actual_to, color, thickness, style, alternate_color);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::fill_path(Path& path, Color color, Painter::WindingRule rule)
|
|
{
|
|
Detail::fill_path<Detail::FillPathMode::AllowFloatingPoints>(*this, path, color, rule);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::stroke_path(Path const& path, Color color, float thickness)
|
|
{
|
|
FloatPoint cursor;
|
|
|
|
for (auto& segment : path.segments()) {
|
|
switch (segment.type()) {
|
|
case Segment::Type::Invalid:
|
|
VERIFY_NOT_REACHED();
|
|
case Segment::Type::MoveTo:
|
|
cursor = segment.point();
|
|
break;
|
|
case Segment::Type::LineTo:
|
|
draw_line(cursor, segment.point(), color, thickness);
|
|
cursor = segment.point();
|
|
break;
|
|
case Segment::Type::QuadraticBezierCurveTo: {
|
|
auto& through = static_cast<QuadraticBezierCurveSegment const&>(segment).through();
|
|
draw_quadratic_bezier_curve(through, cursor, segment.point(), color, thickness);
|
|
cursor = segment.point();
|
|
break;
|
|
}
|
|
case Segment::Type::CubicBezierCurveTo: {
|
|
auto& curve = static_cast<CubicBezierCurveSegment const&>(segment);
|
|
auto& through_0 = curve.through_0();
|
|
auto& through_1 = curve.through_1();
|
|
draw_cubic_bezier_curve(through_0, through_1, cursor, segment.point(), color, thickness);
|
|
cursor = segment.point();
|
|
break;
|
|
}
|
|
case Segment::Type::EllipticalArcTo:
|
|
auto& arc = static_cast<EllipticalArcSegment const&>(segment);
|
|
draw_elliptical_arc(cursor, segment.point(), arc.center(), arc.radii(), arc.x_axis_rotation(), arc.theta_1(), arc.theta_delta(), color, thickness);
|
|
cursor = segment.point();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_elliptical_arc(FloatPoint const& p1, FloatPoint const& p2, FloatPoint const& center, FloatPoint const& radii, float x_axis_rotation, float theta_1, float theta_delta, Color color, float thickness, Painter::LineStyle style)
|
|
{
|
|
Gfx::Painter::for_each_line_segment_on_elliptical_arc(p1, p2, center, radii, x_axis_rotation, theta_1, theta_delta, [&](FloatPoint const& fp1, FloatPoint const& fp2) {
|
|
draw_line(fp1, fp2, color, thickness, style);
|
|
});
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_quadratic_bezier_curve(FloatPoint const& control_point, FloatPoint const& p1, FloatPoint const& p2, Color color, float thickness, Painter::LineStyle style)
|
|
{
|
|
Gfx::Painter::for_each_line_segment_on_bezier_curve(control_point, p1, p2, [&](FloatPoint const& fp1, FloatPoint const& fp2) {
|
|
draw_line(fp1, fp2, color, thickness, style);
|
|
});
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_cubic_bezier_curve(FloatPoint const& control_point_0, FloatPoint const& control_point_1, FloatPoint const& p1, FloatPoint const& p2, Color color, float thickness, Painter::LineStyle style)
|
|
{
|
|
Gfx::Painter::for_each_line_segment_on_cubic_bezier_curve(control_point_0, control_point_1, p1, p2, [&](FloatPoint const& fp1, FloatPoint const& fp2) {
|
|
draw_line(fp1, fp2, color, thickness, style);
|
|
});
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::draw_ellipse(IntRect const& a_rect, Color color, int thickness)
|
|
{
|
|
// FIXME: Come up with an allocation-free version of this!
|
|
// Using draw_line() for segments of an ellipse was attempted but gave really poor results :^(
|
|
// There probably is a way to adjust the fill of draw_ellipse_part() to do this, but gettting it rendering correctly is tricky.
|
|
// The outline of the steps required to paint it efficiently is:
|
|
// - Paint the outer ellipse without the fill (from the fill() lambda in draw_ellipse_part())
|
|
// - Paint the inner ellipse, but in the set_pixel() invert the alpha values
|
|
// - Somehow fill in the gap between the two ellipses (the tricky part to get right)
|
|
// - Have to avoid overlapping pixels and accidentally painting over some of the edge pixels
|
|
|
|
auto color_no_alpha = color;
|
|
color_no_alpha.set_alpha(255);
|
|
auto outline_ellipse_bitmap = ({
|
|
auto bitmap = Gfx::Bitmap::try_create(BitmapFormat::BGRA8888, a_rect.size());
|
|
if (bitmap.is_error())
|
|
return warnln("Failed to allocate temporary bitmap for antialiased outline ellipse!");
|
|
bitmap.release_value();
|
|
});
|
|
|
|
auto outer_rect = a_rect;
|
|
outer_rect.set_location({ 0, 0 });
|
|
auto inner_rect = outer_rect.shrunken(thickness * 2, thickness * 2);
|
|
Gfx::Painter painter { outline_ellipse_bitmap };
|
|
AntiAliasingPainter aa_painter { painter };
|
|
aa_painter.fill_ellipse(outer_rect, color_no_alpha);
|
|
aa_painter.fill_ellipse(inner_rect, color_no_alpha, BlendMode::AlphaSubtract);
|
|
m_underlying_painter.blit(a_rect.location(), outline_ellipse_bitmap, outline_ellipse_bitmap->rect(), color.alpha() / 255.);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::fill_circle(IntPoint const& center, int radius, Color color, BlendMode blend_mode)
|
|
{
|
|
if (radius <= 0)
|
|
return;
|
|
draw_ellipse_part(center, radius, radius, color, false, {}, blend_mode);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::fill_ellipse(IntRect const& a_rect, Color color, BlendMode blend_mode)
|
|
{
|
|
auto center = a_rect.center();
|
|
auto radius_a = a_rect.width() / 2;
|
|
auto radius_b = a_rect.height() / 2;
|
|
if (radius_a <= 0 || radius_b <= 0)
|
|
return;
|
|
if (radius_a == radius_b)
|
|
return fill_circle(center, radius_a, color, blend_mode);
|
|
auto x_paint_range = draw_ellipse_part(center, radius_a, radius_b, color, false, {}, blend_mode);
|
|
// FIXME: This paints some extra fill pixels that are clipped
|
|
draw_ellipse_part(center, radius_b, radius_a, color, true, x_paint_range, blend_mode);
|
|
}
|
|
|
|
Gfx::AntiAliasingPainter::Range Gfx::AntiAliasingPainter::draw_ellipse_part(
|
|
IntPoint center, int radius_a, int radius_b, Color color, bool flip_x_and_y, Optional<Range> x_clip, BlendMode blend_mode)
|
|
{
|
|
/*
|
|
Algorithm from: https://cs.uwaterloo.ca/research/tr/1984/CS-84-38.pdf
|
|
|
|
This method can draw a whole circle with a whole circle in one call using
|
|
8-way symmetry, or an ellipse in two calls using 4-way symmetry.
|
|
*/
|
|
|
|
center *= m_underlying_painter.scale();
|
|
radius_a *= m_underlying_painter.scale();
|
|
radius_b *= m_underlying_painter.scale();
|
|
|
|
// If this is a ellipse everything can be drawn in one pass with 8 way symmetry
|
|
bool const is_circle = radius_a == radius_b;
|
|
|
|
// These happen to be the same here, but are treated separately in the paper:
|
|
// intensity is the fill alpha
|
|
int const intensity = 255;
|
|
// 0 to subpixel_resolution is the range of alpha values for the circle edges
|
|
int const subpixel_resolution = intensity;
|
|
|
|
// Current pixel address
|
|
int i = 0;
|
|
int q = radius_b;
|
|
|
|
// 1st and 2nd order differences of y
|
|
int delta_y = 0;
|
|
int delta2_y = 0;
|
|
|
|
int const a_squared = radius_a * radius_a;
|
|
int const b_squared = radius_b * radius_b;
|
|
|
|
// Exact and predicted values of f(i) -- the ellipse equation scaled by subpixel_resolution
|
|
int y = subpixel_resolution * radius_b;
|
|
int y_hat = 0;
|
|
|
|
// The value of f(i)*f(i)
|
|
int f_squared = y * y;
|
|
|
|
// 1st and 2nd order differences of f(i)*f(i)
|
|
int delta_f_squared = -(static_cast<int64_t>(b_squared) * subpixel_resolution * subpixel_resolution) / a_squared;
|
|
int delta2_f_squared = 2 * delta_f_squared;
|
|
|
|
// edge_intersection_area/subpixel_resolution = percentage of pixel intersected by circle
|
|
// (aka the alpha for the pixel)
|
|
int edge_intersection_area = 0;
|
|
int old_area = edge_intersection_area;
|
|
|
|
auto predict = [&] {
|
|
delta_y += delta2_y;
|
|
// y_hat is the predicted value of f(i)
|
|
y_hat = y + delta_y;
|
|
};
|
|
|
|
auto minimize = [&] {
|
|
// Initialize the minimization
|
|
delta_f_squared += delta2_f_squared;
|
|
f_squared += delta_f_squared;
|
|
|
|
int min_squared_error = y_hat * y_hat - f_squared;
|
|
int prediction_overshot = 1;
|
|
y = y_hat;
|
|
|
|
// Force error negative
|
|
if (min_squared_error > 0) {
|
|
min_squared_error = -min_squared_error;
|
|
prediction_overshot = -1;
|
|
}
|
|
|
|
// Minimize
|
|
int previous_error = min_squared_error;
|
|
while (min_squared_error < 0) {
|
|
y += prediction_overshot;
|
|
previous_error = min_squared_error;
|
|
min_squared_error += y + y - prediction_overshot;
|
|
}
|
|
|
|
if (min_squared_error + previous_error > 0)
|
|
y -= prediction_overshot;
|
|
};
|
|
|
|
auto correct = [&] {
|
|
int error = y - y_hat;
|
|
|
|
// FIXME: The alpha values seem too low, which makes things look
|
|
// overly pointy. This fixes that, though there's probably a better
|
|
// solution to be found. (This issue seems to exist in the base algorithm)
|
|
error /= 4;
|
|
|
|
delta2_y += error;
|
|
delta_y += error;
|
|
};
|
|
|
|
int min_paint_x = NumericLimits<int>::max();
|
|
int max_paint_x = NumericLimits<int>::min();
|
|
auto pixel = [&](int x, int y, int alpha) {
|
|
if (alpha <= 0 || alpha > 255)
|
|
return;
|
|
if (flip_x_and_y)
|
|
swap(x, y);
|
|
if (x_clip.has_value() && x_clip->contains_inclusive(x))
|
|
return;
|
|
min_paint_x = min(x, min_paint_x);
|
|
max_paint_x = max(x, max_paint_x);
|
|
alpha = (alpha * color.alpha()) / 255;
|
|
if (blend_mode == BlendMode::AlphaSubtract)
|
|
alpha = ~alpha;
|
|
auto pixel_color = color;
|
|
pixel_color.set_alpha(alpha);
|
|
m_underlying_painter.set_pixel(center + IntPoint { x, y }, pixel_color, blend_mode == BlendMode::Normal);
|
|
};
|
|
|
|
auto fill = [&](int x, int ymax, int ymin, int alpha) {
|
|
while (ymin <= ymax) {
|
|
pixel(x, ymin, alpha);
|
|
ymin += 1;
|
|
}
|
|
};
|
|
|
|
auto symmetric_pixel = [&](int x, int y, int alpha) {
|
|
pixel(x, y, alpha);
|
|
pixel(x, -y - 1, alpha);
|
|
pixel(-x - 1, -y - 1, alpha);
|
|
pixel(-x - 1, y, alpha);
|
|
if (is_circle) {
|
|
pixel(y, x, alpha);
|
|
pixel(y, -x - 1, alpha);
|
|
pixel(-y - 1, -x - 1, alpha);
|
|
pixel(-y - 1, x, alpha);
|
|
}
|
|
};
|
|
|
|
// These are calculated incrementally (as it is possibly a tiny bit faster)
|
|
int ib_squared = 0;
|
|
int qa_squared = q * a_squared;
|
|
|
|
auto in_symmetric_region = [&] {
|
|
// Main fix two stop cond here
|
|
return is_circle ? i < q : ib_squared < qa_squared;
|
|
};
|
|
|
|
// Draws a 8 octants for a circle or 4 quadrants for a (partial) ellipse
|
|
while (in_symmetric_region()) {
|
|
predict();
|
|
minimize();
|
|
correct();
|
|
old_area = edge_intersection_area;
|
|
edge_intersection_area += delta_y;
|
|
if (edge_intersection_area >= 0) {
|
|
// Single pixel on perimeter
|
|
symmetric_pixel(i, q, (edge_intersection_area + old_area) / 2);
|
|
fill(i, q - 1, -q, intensity);
|
|
fill(-i - 1, q - 1, -q, intensity);
|
|
} else {
|
|
// Two pixels on perimeter
|
|
edge_intersection_area += subpixel_resolution;
|
|
symmetric_pixel(i, q, old_area / 2);
|
|
q -= 1;
|
|
qa_squared -= a_squared;
|
|
fill(i, q - 1, -q, intensity);
|
|
fill(-i - 1, q - 1, -q, intensity);
|
|
if (!is_circle || in_symmetric_region()) {
|
|
symmetric_pixel(i, q, (edge_intersection_area + subpixel_resolution) / 2);
|
|
if (is_circle) {
|
|
fill(q, i - 1, -i, intensity);
|
|
fill(-q - 1, i - 1, -i, intensity);
|
|
}
|
|
} else {
|
|
edge_intersection_area += subpixel_resolution;
|
|
}
|
|
}
|
|
i += 1;
|
|
ib_squared += b_squared;
|
|
}
|
|
|
|
if (is_circle) {
|
|
int alpha = edge_intersection_area / 2;
|
|
pixel(q, q, alpha);
|
|
pixel(-q - 1, q, alpha);
|
|
pixel(-q - 1, -q - 1, alpha);
|
|
pixel(q, -q - 1, alpha);
|
|
}
|
|
|
|
return Range { min_paint_x, max_paint_x };
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::fill_rect_with_rounded_corners(IntRect const& a_rect, Color color, int radius)
|
|
{
|
|
fill_rect_with_rounded_corners(a_rect, color, radius, radius, radius, radius);
|
|
}
|
|
|
|
void Gfx::AntiAliasingPainter::fill_rect_with_rounded_corners(IntRect const& a_rect, Color color, int top_left_radius, int top_right_radius, int bottom_right_radius, int bottom_left_radius)
|
|
{
|
|
if (!top_left_radius && !top_right_radius && !bottom_right_radius && !bottom_left_radius)
|
|
return m_underlying_painter.fill_rect(a_rect, color);
|
|
|
|
if (color.alpha() == 0)
|
|
return;
|
|
|
|
IntPoint top_left_corner {
|
|
a_rect.x() + top_left_radius,
|
|
a_rect.y() + top_left_radius,
|
|
};
|
|
IntPoint top_right_corner {
|
|
a_rect.x() + a_rect.width() - top_right_radius,
|
|
a_rect.y() + top_right_radius,
|
|
};
|
|
IntPoint bottom_left_corner {
|
|
a_rect.x() + bottom_left_radius,
|
|
a_rect.y() + a_rect.height() - bottom_left_radius
|
|
};
|
|
IntPoint bottom_right_corner {
|
|
a_rect.x() + a_rect.width() - bottom_right_radius,
|
|
a_rect.y() + a_rect.height() - bottom_right_radius
|
|
};
|
|
|
|
IntRect top_rect {
|
|
a_rect.x() + top_left_radius,
|
|
a_rect.y(),
|
|
a_rect.width() - top_left_radius - top_right_radius,
|
|
top_left_radius
|
|
};
|
|
IntRect right_rect {
|
|
a_rect.x() + a_rect.width() - top_right_radius,
|
|
a_rect.y() + top_right_radius,
|
|
top_right_radius,
|
|
a_rect.height() - top_right_radius - bottom_right_radius
|
|
};
|
|
IntRect bottom_rect {
|
|
a_rect.x() + bottom_left_radius,
|
|
a_rect.y() + a_rect.height() - bottom_right_radius,
|
|
a_rect.width() - bottom_left_radius - bottom_right_radius,
|
|
bottom_right_radius
|
|
};
|
|
IntRect left_rect {
|
|
a_rect.x(),
|
|
a_rect.y() + top_left_radius,
|
|
bottom_left_radius,
|
|
a_rect.height() - top_left_radius - bottom_left_radius
|
|
};
|
|
|
|
IntRect inner = {
|
|
left_rect.x() + left_rect.width(),
|
|
left_rect.y(),
|
|
a_rect.width() - left_rect.width() - right_rect.width(),
|
|
a_rect.height() - top_rect.height() - bottom_rect.height()
|
|
};
|
|
|
|
m_underlying_painter.fill_rect(top_rect, color);
|
|
m_underlying_painter.fill_rect(right_rect, color);
|
|
m_underlying_painter.fill_rect(bottom_rect, color);
|
|
m_underlying_painter.fill_rect(left_rect, color);
|
|
m_underlying_painter.fill_rect(inner, color);
|
|
|
|
enum class CornerClip {
|
|
TopLeft,
|
|
BottomLeft,
|
|
TopRight,
|
|
BottomRight
|
|
};
|
|
|
|
auto fill_corner = [&](auto const& point, int radius, CornerClip clip) {
|
|
PainterStateSaver save { m_underlying_painter };
|
|
auto corner_clip = IntRect::from_two_points(point, [&] {
|
|
switch (clip) {
|
|
case CornerClip::TopLeft:
|
|
return point.translated(-radius, -radius);
|
|
case CornerClip::BottomLeft:
|
|
return point.translated(-radius, radius);
|
|
case CornerClip::TopRight:
|
|
return point.translated(radius, -radius);
|
|
case CornerClip::BottomRight:
|
|
return point.translated(radius, radius);
|
|
default:
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
}());
|
|
m_underlying_painter.add_clip_rect(corner_clip);
|
|
fill_circle(point, radius, color);
|
|
};
|
|
|
|
if (top_left_radius)
|
|
fill_corner(top_left_corner, top_left_radius, CornerClip::TopLeft);
|
|
if (top_right_radius)
|
|
fill_corner(top_right_corner, top_right_radius, CornerClip::TopRight);
|
|
if (bottom_left_radius)
|
|
fill_corner(bottom_left_corner, bottom_left_radius, CornerClip::BottomLeft);
|
|
if (bottom_right_radius)
|
|
fill_corner(bottom_right_corner, bottom_right_radius, CornerClip::BottomRight);
|
|
}
|