canvas: draw circle using cairo (#10007)

the circle is now antialiased using cairo.
This commit is contained in:
Subhraman Sarkar 2025-03-11 01:34:49 +05:30 committed by GitHub
parent 55c228fcbb
commit bd6f250c12
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 95 additions and 27 deletions

View File

@ -18,20 +18,23 @@
#define _GUI_RADIO_BG COLOR
[circle]
x = 10
y = 10
radius = 10
y = 13
radius = 11
border_thickness = 2
border_color = {COLOR}
fill_color = "28, 45, 64, 255"
[/circle]
#enddef
#define _GUI_RADIO_DOT IPF
[image]
x = 6
y = 6
name = "buttons/modern/dot.png{IPF}"
[/image]
#define _GUI_RADIO_DOT COLOR
[circle]
x = 10
y = 13
radius = 5
border_thickness = 0
border_color = "0, 0, 0, 0"
fill_color = {COLOR}
[/circle]
#enddef
#define _GUI_RESOLUTION RESOLUTION WIDTH HEIGHT EXTRA_WIDTH FONT_SIZE
@ -96,7 +99,7 @@
[draw]
{_GUI_RADIO_BG ({GUI__COLOR_TOGGLE_ENABLED})}
{_GUI_RADIO_DOT ()}
{_GUI_RADIO_DOT ("255, 225, 104, 255")}
{_GUI_TEXT ({EXTRA_WIDTH}) ({FONT_SIZE}) ({GUI__FONT_COLOR_ENABLED__TITLE}) }
[/draw]
@ -108,7 +111,7 @@
[draw]
{_GUI_RADIO_BG ({GUI__COLOR_TOGGLE_DISABLED})}
{_GUI_RADIO_DOT "~GS()"}
{_GUI_RADIO_DOT ("179, 179, 179, 255")}
{_GUI_TEXT ({EXTRA_WIDTH}) ({FONT_SIZE}) ({GUI__FONT_COLOR_DISABLED__TITLE}) }
[/draw]
@ -120,7 +123,7 @@
[draw]
{_GUI_RADIO_BG ({GUI__FONT_COLOR_ENABLED__BRIGHT})}
{_GUI_RADIO_DOT ()}
{_GUI_RADIO_DOT ("255, 225, 104, 255")}
{_GUI_TEXT ({EXTRA_WIDTH}) ({FONT_SIZE}) ({GUI__FONT_COLOR_ENABLED__TITLE}) }
[/draw]
@ -136,7 +139,7 @@
id = "radio"
description = "Radio button."
{_GUI_RESOLUTION () 32 24 25 ({GUI_FONT_SIZE_SMALL}) }
{_GUI_RESOLUTION () 36 26 25 ({GUI_FONT_SIZE_SMALL}) }
[/toggle_button_definition]
@ -145,7 +148,7 @@
id = "radio_no_label"
description = "Radio button."
{_GUI_RESOLUTION () 32 24 25 ({GUI_FONT_SIZE_SMALL}) }
{_GUI_RESOLUTION () 36 26 25 ({GUI_FONT_SIZE_SMALL}) }
[/toggle_button_definition]

Binary file not shown.

Before

Width:  |  Height:  |  Size: 898 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.3 KiB

View File

@ -15,14 +15,17 @@
#include "draw.hpp"
#include "color.hpp"
#include "font/cairo.hpp"
#include "log.hpp"
#include "sdl/rect.hpp"
#include "sdl/texture.hpp"
#include "sdl/utils.hpp" // sdl::runtime_at_least
#include "video.hpp"
#include <boost/math/constants/constants.hpp>
#include <SDL2/SDL_rect.h>
#include <SDL2/SDL_render.h>
#include <cairo.h>
static lg::log_domain log_draw("draw");
#define DBG_D LOG_STREAM(debug, log_draw)
@ -308,6 +311,67 @@ void draw::disc(int cx, int cy, int r, uint8_t octants)
}
}
/********************/
/* Cairo primitives */
/********************/
void draw::cairo_circle(int cx, int cy, int r, const color_t& c, int thickness)
{
if (r <= 0) {
return;
}
int size = 2*r;
surface sdl_surf(size, size);
auto cairo_surface = cairo::create_surface(
reinterpret_cast<uint8_t*>(sdl_surf->pixels), ::point(sdl_surf->w, sdl_surf->h));
auto cairo_context = cairo::create_context(cairo_surface);
cairo_t* ctx = cairo_context.get();
cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
cairo_set_source_rgba(ctx, 0.0, 0.0, 0.0, 0.0);
cairo_paint(ctx);
cairo_set_line_width(ctx, thickness);
cairo_set_source_rgba(ctx,
c.r / 255.0,
c.g / 255.0,
c.b / 255.0,
c.a / 255.0
);
cairo_arc(ctx, r, r, r-thickness, 0, 2*boost::math::constants::pi<double>());
cairo_stroke(ctx);
draw::blit(texture(sdl_surf), ::rect(cx-r, cy-r, size, size));
}
void draw::cairo_disc(int cx, int cy, int r, const color_t& c)
{
if (r <= 0) {
return;
}
int size = 2*r;
surface sdl_surf(size, size);
auto cairo_surface = cairo::create_surface(
reinterpret_cast<uint8_t*>(sdl_surf->pixels), ::point(sdl_surf->w, sdl_surf->h));
auto cairo_context = cairo::create_context(cairo_surface);
cairo_t* ctx = cairo_context.get();
cairo_set_antialias(ctx, CAIRO_ANTIALIAS_BEST);
cairo_set_source_rgba(ctx,
c.r / 255.0,
c.g / 255.0,
c.b / 255.0,
c.a / 255.0
);
cairo_arc(ctx, r, r, r, 0, 2*2*boost::math::constants::pi<double>());
cairo_fill(ctx);
draw::blit(texture(sdl_surf), ::rect(cx-r, cy-r, size, size));
}
/*******************/
/* texture drawing */

View File

@ -32,10 +32,10 @@
#include "sdl/rect.hpp"
#include "sdl/texture.hpp"
#include <array>
#include <vector>
#include <SDL2/SDL_render.h>
#include <array>
struct color_t;
@ -210,6 +210,12 @@ void circle(int x, int y, int r, uint8_t octants = 0xff);
void disc(int x, int y, int r, const color_t& c, uint8_t octants = 0xff);
void disc(int x, int y, int r, uint8_t octants = 0xff);
/** Draw outline of circle using Cairo */
void cairo_circle(int cx, int cy, int r, const color_t& c, int thickness);
/** Draw filled circle using Cairo */
void cairo_disc(int cx, int cy, int r, const color_t& c);
/*******************/
/* texture drawing */

View File

@ -27,7 +27,7 @@ using context_ptr = std::unique_ptr<cairo_t, void(*)(cairo_t*)>;
/** Color format for cairo surfaces. Should be equivalent to the format used by SDL. */
constexpr cairo_format_t format = CAIRO_FORMAT_ARGB32;
surface_ptr create_surface(uint8_t* buffer, const point& size)
inline surface_ptr create_surface(uint8_t* buffer, const point& size)
{
const auto& [width, height] = size;
const int stride = cairo_format_stride_for_width(format, width);
@ -38,7 +38,7 @@ surface_ptr create_surface(uint8_t* buffer, const point& size)
};
}
context_ptr create_context(const surface_ptr& surf)
inline context_ptr create_context(const surface_ptr& surf)
{
return {
cairo_create(surf.get()),

View File

@ -36,13 +36,12 @@
#include "picture.hpp"
#include "sdl/point.hpp"
#include "sdl/rect.hpp"
#include "sdl/surface.hpp"
#include "sdl/texture.hpp"
#include "sdl/utils.hpp" // blur_surface
#include "video.hpp" // read_pixels_low_res, only used for blurring
#include "wml_exception.hpp"
#include <iostream>
namespace gui2
{
@ -219,22 +218,18 @@ void circle_shape::draw(wfl::map_formula_callable& variables)
* silly unless there has been a resize. So to optimize we should use an
* extra flag or do the calculation in a separate routine.
*/
const int x = x_(variables);
const int y = y_(variables);
const unsigned radius = radius_(variables);
DBG_GUI_D << "Circle: drawn at " << x << ',' << y << " radius " << radius << ".";
const color_t fill_color = fill_color_(variables);
if(!fill_color.null() && radius) {
draw::disc(x, y, radius, fill_color);
if (!fill_color.null()) {
draw::cairo_disc(x, y, radius, fill_color);
}
const color_t border_color = border_color_(variables);
for(unsigned int i = 0; i < border_thickness_; i++) {
draw::circle(x, y, radius - i, border_color);
}
draw::cairo_circle(x, y, radius, border_color, border_thickness_);
DBG_GUI_D << "Circle: drawn at " << x << ',' << y << " radius " << radius << ".";
}
/***** ***** ***** ***** ***** IMAGE ***** ***** ***** ***** *****/