ES-DE/3rdparty/plutovg/plutovg-paint.c
Leon Styhre e5b9b53006 Squashed 'external/lunasvg/' changes from ead790126..be5d0977b
be5d0977b Fix possible memory leaks
62e0e4b31 Optimize plutovg memory allocations
587539885 Support graphical empty files #109

git-subtree-dir: external/lunasvg
git-subtree-split: be5d0977b393055f89be70323ac9b020ef3ab657
2022-10-26 20:03:31 +02:00

266 lines
8.4 KiB
C

#include "plutovg-private.h"
void plutovg_color_init_rgb(plutovg_color_t* color, double r, double g, double b)
{
plutovg_color_init_rgba(color, r, g, b, 1.0);
}
void plutovg_color_init_rgba(plutovg_color_t* color, double r, double g, double b, double a)
{
color->r = plutovg_clamp(r, 0.0, 1.0);
color->g = plutovg_clamp(g, 0.0, 1.0);
color->b = plutovg_clamp(b, 0.0, 1.0);
color->a = plutovg_clamp(a, 0.0, 1.0);
}
void plutovg_gradient_init_linear(plutovg_gradient_t* gradient, double x1, double y1, double x2, double y2)
{
gradient->type = plutovg_gradient_type_linear;
gradient->spread = plutovg_spread_method_pad;
gradient->opacity = 1.0;
plutovg_array_clear(gradient->stops);
plutovg_matrix_init_identity(&gradient->matrix);
plutovg_gradient_set_values_linear(gradient, x1, y1, x2, y2);
}
void plutovg_gradient_init_radial(plutovg_gradient_t* gradient, double cx, double cy, double cr, double fx, double fy, double fr)
{
gradient->type = plutovg_gradient_type_radial;
gradient->spread = plutovg_spread_method_pad;
gradient->opacity = 1.0;
plutovg_array_clear(gradient->stops);
plutovg_matrix_init_identity(&gradient->matrix);
plutovg_gradient_set_values_radial(gradient, cx, cy, cr, fx, fy, fr);
}
void plutovg_gradient_set_spread(plutovg_gradient_t* gradient, plutovg_spread_method_t spread)
{
gradient->spread = spread;
}
plutovg_spread_method_t plutovg_gradient_get_spread(const plutovg_gradient_t* gradient)
{
return gradient->spread;
}
void plutovg_gradient_set_matrix(plutovg_gradient_t* gradient, const plutovg_matrix_t* matrix)
{
gradient->matrix = *matrix;
}
void plutovg_gradient_get_matrix(const plutovg_gradient_t* gradient, plutovg_matrix_t *matrix)
{
*matrix = gradient->matrix;
}
void plutovg_gradient_add_stop_rgb(plutovg_gradient_t* gradient, double offset, double r, double g, double b)
{
plutovg_gradient_add_stop_rgba(gradient, offset, r, g, b, 1.0);
}
void plutovg_gradient_add_stop_rgba(plutovg_gradient_t* gradient, double offset, double r, double g, double b, double a)
{
if(offset < 0.0) offset = 0.0;
if(offset > 1.0) offset = 1.0;
plutovg_array_ensure(gradient->stops, 1);
plutovg_gradient_stop_t* stops = gradient->stops.data;
int nstops = gradient->stops.size;
int i = 0;
for(; i < nstops; i++) {
if(offset < stops[i].offset) {
memmove(&stops[i+1], &stops[i], (size_t)(nstops - i) * sizeof(plutovg_gradient_stop_t));
break;
}
}
plutovg_gradient_stop_t* stop = &stops[i];
stop->offset = offset;
plutovg_color_init_rgba(&stop->color, r, g, b, a);
gradient->stops.size += 1;
}
void plutovg_gradient_add_stop_color(plutovg_gradient_t* gradient, double offset, const plutovg_color_t* color)
{
plutovg_gradient_add_stop_rgba(gradient, offset, color->r, color->g, color->b, color->a);
}
void plutovg_gradient_add_stop(plutovg_gradient_t* gradient, const plutovg_gradient_stop_t* stop)
{
plutovg_gradient_add_stop_rgba(gradient, stop->offset, stop->color.r, stop->color.g, stop->color.b, stop->color.a);
}
void plutovg_gradient_clear_stops(plutovg_gradient_t* gradient)
{
gradient->stops.size = 0;
}
int plutovg_gradient_get_stop_count(const plutovg_gradient_t* gradient)
{
return gradient->stops.size;
}
plutovg_gradient_stop_t* plutovg_gradient_get_stops(const plutovg_gradient_t* gradient)
{
return gradient->stops.data;
}
plutovg_gradient_type_t plutovg_gradient_get_type(const plutovg_gradient_t* gradient)
{
return gradient->type;
}
void plutovg_gradient_get_values_linear(const plutovg_gradient_t* gradient, double* x1, double* y1, double* x2, double* y2)
{
if(x1) *x1 = gradient->values[0];
if(y1) *y1 = gradient->values[1];
if(x2) *x2 = gradient->values[2];
if(y2) *y2 = gradient->values[3];
}
void plutovg_gradient_get_values_radial(const plutovg_gradient_t* gradient, double* cx, double* cy, double* cr, double* fx, double* fy, double* fr)
{
if(cx) *cx = gradient->values[0];
if(cy) *cy = gradient->values[1];
if(cr) *cr = gradient->values[2];
if(fx) *fx = gradient->values[3];
if(fy) *fy = gradient->values[4];
if(fr) *fr = gradient->values[5];
}
void plutovg_gradient_set_values_linear(plutovg_gradient_t* gradient, double x1, double y1, double x2, double y2)
{
gradient->values[0] = x1;
gradient->values[1] = y1;
gradient->values[2] = x2;
gradient->values[3] = y2;
}
void plutovg_gradient_set_values_radial(plutovg_gradient_t* gradient, double cx, double cy, double cr, double fx, double fy, double fr)
{
gradient->values[0] = cx;
gradient->values[1] = cy;
gradient->values[2] = cr;
gradient->values[3] = fx;
gradient->values[4] = fy;
gradient->values[5] = fr;
}
void plutovg_gradient_set_opacity(plutovg_gradient_t* gradient, double opacity)
{
gradient->opacity = plutovg_clamp(opacity, 0.0, 1.0);
}
double plutovg_gradient_get_opacity(const plutovg_gradient_t* gradient)
{
return gradient->opacity;
}
void plutovg_gradient_copy(plutovg_gradient_t* gradient, const plutovg_gradient_t* source)
{
gradient->type = source->type;
gradient->spread = source->spread;
gradient->matrix = source->matrix;
gradient->opacity = source->opacity;
plutovg_array_ensure(gradient->stops, source->stops.size);
memcpy(gradient->values, source->values, sizeof(source->values));
memcpy(gradient->stops.data, source->stops.data, source->stops.size * sizeof(plutovg_gradient_stop_t));
}
void plutovg_gradient_destroy(plutovg_gradient_t* gradient)
{
plutovg_array_destroy(gradient->stops);
}
void plutovg_texture_init(plutovg_texture_t* texture, plutovg_surface_t* surface, plutovg_texture_type_t type)
{
surface = plutovg_surface_reference(surface);
plutovg_surface_destroy(texture->surface);
texture->type = type;
texture->surface = surface;
texture->opacity = 1.0;
plutovg_matrix_init_identity(&texture->matrix);
}
void plutovg_texture_set_type(plutovg_texture_t* texture, plutovg_texture_type_t type)
{
texture->type = type;
}
plutovg_texture_type_t plutovg_texture_get_type(const plutovg_texture_t* texture)
{
return texture->type;
}
void plutovg_texture_set_matrix(plutovg_texture_t* texture, const plutovg_matrix_t* matrix)
{
texture->matrix = *matrix;
}
void plutovg_texture_get_matrix(const plutovg_texture_t* texture, plutovg_matrix_t* matrix)
{
*matrix = texture->matrix;
}
void plutovg_texture_set_surface(plutovg_texture_t* texture, plutovg_surface_t* surface)
{
surface = plutovg_surface_reference(surface);
plutovg_surface_destroy(texture->surface);
texture->surface = surface;
}
plutovg_surface_t* plutovg_texture_get_surface(const plutovg_texture_t* texture)
{
return texture->surface;
}
void plutovg_texture_set_opacity(plutovg_texture_t* texture, double opacity)
{
texture->opacity = plutovg_clamp(opacity, 0.0, 1.0);
}
double plutovg_texture_get_opacity(const plutovg_texture_t* texture)
{
return texture->opacity;
}
void plutovg_texture_copy(plutovg_texture_t* texture, const plutovg_texture_t* source)
{
plutovg_surface_t* surface = plutovg_surface_reference(source->surface);
plutovg_surface_destroy(texture->surface);
texture->type = source->type;
texture->surface = surface;
texture->opacity = source->opacity;
texture->matrix = source->matrix;
}
void plutovg_texture_destroy(plutovg_texture_t* texture)
{
plutovg_surface_destroy(texture->surface);
}
void plutovg_paint_init(plutovg_paint_t* paint)
{
paint->type = plutovg_paint_type_color;
paint->texture.surface = NULL;
plutovg_array_init(paint->gradient.stops);
plutovg_color_init_rgb(&paint->color, 0, 0, 0);
}
void plutovg_paint_destroy(plutovg_paint_t* paint)
{
plutovg_texture_destroy(&paint->texture);
plutovg_gradient_destroy(&paint->gradient);
}
void plutovg_paint_copy(plutovg_paint_t* paint, const plutovg_paint_t* source)
{
paint->type = source->type;
if(source->type == plutovg_paint_type_color)
paint->color = source->color;
else if(source->type == plutovg_paint_type_color)
plutovg_gradient_copy(&paint->gradient, &paint->gradient);
else
plutovg_texture_copy(&paint->texture, &paint->texture);
}