ES-DE/external/lunasvg/3rdparty/plutovg/plutovg.c

496 lines
13 KiB
C

#include "plutovg-private.h"
plutovg_surface_t* plutovg_surface_create(int width, int height)
{
plutovg_surface_t* surface = malloc(sizeof(plutovg_surface_t));
surface->ref = 1;
surface->owndata = 1;
surface->data = calloc(1, (size_t)(width * height * 4));
surface->width = width;
surface->height = height;
surface->stride = width * 4;
return surface;
}
plutovg_surface_t* plutovg_surface_create_for_data(unsigned char* data, int width, int height, int stride)
{
plutovg_surface_t* surface = malloc(sizeof(plutovg_surface_t));
surface->ref = 1;
surface->owndata = 0;
surface->data = data;
surface->width = width;
surface->height = height;
surface->stride = stride;
return surface;
}
plutovg_surface_t* plutovg_surface_reference(plutovg_surface_t* surface)
{
++surface->ref;
return surface;
}
void plutovg_surface_destroy(plutovg_surface_t* surface)
{
if(surface==NULL)
return;
if(--surface->ref==0)
{
if(surface->owndata)
free(surface->data);
free(surface);
}
}
int plutovg_surface_get_reference_count(const plutovg_surface_t* surface)
{
return surface->ref;
}
unsigned char* plutovg_surface_get_data(const plutovg_surface_t* surface)
{
return surface->data;
}
int plutovg_surface_get_width(const plutovg_surface_t* surface)
{
return surface->width;
}
int plutovg_surface_get_height(const plutovg_surface_t* surface)
{
return surface->height;
}
int plutovg_surface_get_stride(const plutovg_surface_t* surface)
{
return surface->stride;
}
plutovg_state_t* plutovg_state_create(void)
{
plutovg_state_t* state = malloc(sizeof(plutovg_state_t));
state->clippath = NULL;
plutovg_paint_init(&state->paint);
plutovg_matrix_init_identity(&state->matrix);
state->winding = plutovg_fill_rule_non_zero;
state->stroke.width = 1.0;
state->stroke.miterlimit = 4.0;
state->stroke.cap = plutovg_line_cap_butt;
state->stroke.join = plutovg_line_join_miter;
state->stroke.dash = NULL;
state->op = plutovg_operator_src_over;
state->opacity = 1.0;
state->next = NULL;
return state;
}
plutovg_state_t* plutovg_state_clone(const plutovg_state_t* state)
{
plutovg_state_t* newstate = plutovg_state_create();
newstate->clippath = plutovg_rle_clone(state->clippath);
plutovg_paint_copy(&newstate->paint, &state->paint);
newstate->matrix = state->matrix;
newstate->winding = state->winding;
newstate->stroke.width = state->stroke.width;
newstate->stroke.miterlimit = state->stroke.miterlimit;
newstate->stroke.cap = state->stroke.cap;
newstate->stroke.join = state->stroke.join;
newstate->stroke.dash = plutovg_dash_clone(state->stroke.dash);
newstate->op = state->op;
newstate->opacity = state->opacity;
newstate->next = NULL;
return newstate;
}
void plutovg_state_destroy(plutovg_state_t* state)
{
plutovg_rle_destroy(state->clippath);
plutovg_paint_destroy(&state->paint);
plutovg_dash_destroy(state->stroke.dash);
free(state);
}
plutovg_t* plutovg_create(plutovg_surface_t* surface)
{
plutovg_t* pluto = malloc(sizeof(plutovg_t));
pluto->ref = 1;
pluto->surface = plutovg_surface_reference(surface);
pluto->state = plutovg_state_create();
pluto->path = plutovg_path_create();
pluto->rle = plutovg_rle_create();
pluto->clippath = NULL;
pluto->clip.x = 0.0;
pluto->clip.y = 0.0;
pluto->clip.w = surface->width;
pluto->clip.h = surface->height;
pluto->outline_data = NULL;
pluto->outline_size = 0;
return pluto;
}
plutovg_t* plutovg_reference(plutovg_t* pluto)
{
++pluto->ref;
return pluto;
}
void plutovg_destroy(plutovg_t* pluto)
{
if(pluto==NULL)
return;
if(--pluto->ref==0)
{
while(pluto->state)
{
plutovg_state_t* state = pluto->state;
pluto->state = state->next;
plutovg_state_destroy(state);
}
plutovg_surface_destroy(pluto->surface);
plutovg_path_destroy(pluto->path);
plutovg_rle_destroy(pluto->rle);
plutovg_rle_destroy(pluto->clippath);
free(pluto->outline_data);
free(pluto);
}
}
int plutovg_get_reference_count(const plutovg_t* pluto)
{
return pluto->ref;
}
void plutovg_save(plutovg_t* pluto)
{
plutovg_state_t* newstate = plutovg_state_clone(pluto->state);
newstate->next = pluto->state;
pluto->state = newstate;
}
void plutovg_restore(plutovg_t* pluto)
{
plutovg_state_t* oldstate = pluto->state;
pluto->state = oldstate->next;
plutovg_state_destroy(oldstate);
}
plutovg_color_t* plutovg_set_rgb(plutovg_t* pluto, double r, double g, double b)
{
return plutovg_set_rgba(pluto, r, g, b, 1.0);
}
plutovg_color_t* plutovg_set_rgba(plutovg_t* pluto, double r, double g, double b, double a)
{
plutovg_paint_t* paint = &pluto->state->paint;
paint->type = plutovg_paint_type_color;
plutovg_color_init_rgba(&paint->color, r, g, b, a);
return &paint->color;
}
plutovg_color_t* plutovg_set_color(plutovg_t* pluto, const plutovg_color_t* color)
{
return plutovg_set_rgba(pluto, color->r, color->g, color->b, color->a);
}
plutovg_gradient_t* plutovg_set_linear_gradient(plutovg_t* pluto, double x1, double y1, double x2, double y2)
{
plutovg_paint_t* paint = &pluto->state->paint;
paint->type = plutovg_paint_type_gradient;
plutovg_gradient_init_linear(&paint->gradient, x1, y1, x2, y2);
return &paint->gradient;
}
plutovg_gradient_t* plutovg_set_radial_gradient(plutovg_t* pluto, double cx, double cy, double cr, double fx, double fy, double fr)
{
plutovg_paint_t* paint = &pluto->state->paint;
paint->type = plutovg_paint_type_gradient;
plutovg_gradient_init_radial(&paint->gradient, cx, cy, cr, fx, fy, fr);
return &paint->gradient;
}
plutovg_texture_t* plutovg_set_texture_surface(plutovg_t* pluto, plutovg_surface_t* surface, double x, double y)
{
plutovg_texture_t* texture = plutovg_set_texture(pluto, surface, plutovg_texture_type_plain);
plutovg_matrix_init_translate(&texture->matrix, x, y);
return texture;
}
plutovg_texture_t* plutovg_set_texture(plutovg_t* pluto, plutovg_surface_t* surface, plutovg_texture_type_t type)
{
plutovg_paint_t* paint = &pluto->state->paint;
paint->type = plutovg_paint_type_texture;
plutovg_texture_init(&paint->texture, surface, type);
return &paint->texture;
}
void plutovg_set_operator(plutovg_t* pluto, plutovg_operator_t op)
{
pluto->state->op = op;
}
void plutovg_set_opacity(plutovg_t* pluto, double opacity)
{
pluto->state->opacity = opacity;
}
void plutovg_set_fill_rule(plutovg_t* pluto, plutovg_fill_rule_t fill_rule)
{
pluto->state->winding = fill_rule;
}
plutovg_operator_t plutovg_get_operator(const plutovg_t* pluto)
{
return pluto->state->op;
}
double plutovg_get_opacity(const plutovg_t* pluto)
{
return pluto->state->opacity;
}
plutovg_fill_rule_t plutovg_get_fill_rule(const plutovg_t* pluto)
{
return pluto->state->winding;
}
void plutovg_set_line_width(plutovg_t* pluto, double width)
{
pluto->state->stroke.width = width;
}
void plutovg_set_line_cap(plutovg_t* pluto, plutovg_line_cap_t cap)
{
pluto->state->stroke.cap = cap;
}
void plutovg_set_line_join(plutovg_t* pluto, plutovg_line_join_t join)
{
pluto->state->stroke.join = join;
}
void plutovg_set_miter_limit(plutovg_t* pluto, double limit)
{
pluto->state->stroke.miterlimit = limit;
}
void plutovg_set_dash(plutovg_t* pluto, double offset, const double* data, int size)
{
plutovg_dash_destroy(pluto->state->stroke.dash);
pluto->state->stroke.dash = plutovg_dash_create(offset, data, size);
}
double plutovg_get_line_width(const plutovg_t* pluto)
{
return pluto->state->stroke.width;
}
plutovg_line_cap_t plutovg_get_line_cap(const plutovg_t* pluto)
{
return pluto->state->stroke.cap;
}
plutovg_line_join_t plutovg_get_line_join(const plutovg_t* pluto)
{
return pluto->state->stroke.join;
}
double plutovg_get_miter_limit(const plutovg_t* pluto)
{
return pluto->state->stroke.miterlimit;
}
void plutovg_translate(plutovg_t* pluto, double x, double y)
{
plutovg_matrix_translate(&pluto->state->matrix, x, y);
}
void plutovg_scale(plutovg_t* pluto, double x, double y)
{
plutovg_matrix_scale(&pluto->state->matrix, x, y);
}
void plutovg_rotate(plutovg_t* pluto, double radians, double x, double y)
{
plutovg_matrix_rotate(&pluto->state->matrix, radians, x, y);
}
void plutovg_transform(plutovg_t* pluto, const plutovg_matrix_t* matrix)
{
plutovg_matrix_multiply(&pluto->state->matrix, matrix, &pluto->state->matrix);
}
void plutovg_set_matrix(plutovg_t* pluto, const plutovg_matrix_t* matrix)
{
pluto->state->matrix = *matrix;
}
void plutovg_identity_matrix(plutovg_t* pluto)
{
plutovg_matrix_init_identity(&pluto->state->matrix);
}
void plutovg_get_matrix(const plutovg_t* pluto, plutovg_matrix_t* matrix)
{
*matrix = pluto->state->matrix;
}
void plutovg_move_to(plutovg_t* pluto, double x, double y)
{
plutovg_path_move_to(pluto->path, x, y);
}
void plutovg_line_to(plutovg_t* pluto, double x, double y)
{
plutovg_path_line_to(pluto->path, x, y);
}
void plutovg_quad_to(plutovg_t* pluto, double x1, double y1, double x2, double y2)
{
plutovg_path_quad_to(pluto->path, x1, y1, x2, y2);
}
void plutovg_cubic_to(plutovg_t* pluto, double x1, double y1, double x2, double y2, double x3, double y3)
{
plutovg_path_cubic_to(pluto->path, x1, y1, x2, y2, x3, y3);
}
void plutovg_rel_move_to(plutovg_t* pluto, double x, double y)
{
plutovg_path_rel_move_to(pluto->path, x, y);
}
void plutovg_rel_line_to(plutovg_t* pluto, double x, double y)
{
plutovg_path_rel_line_to(pluto->path, x, y);
}
void plutovg_rel_quad_to(plutovg_t* pluto, double x1, double y1, double x2, double y2)
{
plutovg_path_rel_quad_to(pluto->path, x1, y1, x2, y2);
}
void plutovg_rel_cubic_to(plutovg_t* pluto, double x1, double y1, double x2, double y2, double x3, double y3)
{
plutovg_path_rel_cubic_to(pluto->path, x1, y1, x2, y2, x3, y3);
}
void plutovg_rect(plutovg_t* pluto, double x, double y, double w, double h)
{
plutovg_path_add_rect(pluto->path, x, y, w, h);
}
void plutovg_round_rect(plutovg_t* pluto, double x, double y, double w, double h, double rx, double ry)
{
plutovg_path_add_round_rect(pluto->path, x, y, w, h, rx, ry);
}
void plutovg_ellipse(plutovg_t* pluto, double cx, double cy, double rx, double ry)
{
plutovg_path_add_ellipse(pluto->path, cx, cy, rx, ry);
}
void plutovg_circle(plutovg_t* pluto, double cx, double cy, double r)
{
plutovg_ellipse(pluto, cx, cy, r, r);
}
void plutovg_add_path(plutovg_t* pluto, const plutovg_path_t* path)
{
plutovg_path_add_path(pluto->path, path, NULL);
}
void plutovg_new_path(plutovg_t* pluto)
{
plutovg_path_clear(pluto->path);
}
void plutovg_close_path(plutovg_t* pluto)
{
plutovg_path_close(pluto->path);
}
plutovg_path_t* plutovg_get_path(const plutovg_t* pluto)
{
return pluto->path;
}
void plutovg_fill(plutovg_t* pluto)
{
plutovg_fill_preserve(pluto);
plutovg_new_path(pluto);
}
void plutovg_stroke(plutovg_t* pluto)
{
plutovg_stroke_preserve(pluto);
plutovg_new_path(pluto);
}
void plutovg_clip(plutovg_t* pluto)
{
plutovg_clip_preserve(pluto);
plutovg_new_path(pluto);
}
void plutovg_paint(plutovg_t* pluto)
{
plutovg_state_t* state = pluto->state;
if(state->clippath==NULL && pluto->clippath==NULL)
{
plutovg_path_t* path = plutovg_path_create();
plutovg_path_add_rect(path, pluto->clip.x, pluto->clip.y, pluto->clip.w, pluto->clip.h);
plutovg_matrix_t matrix;
plutovg_matrix_init_identity(&matrix);
pluto->clippath = plutovg_rle_create();
plutovg_rle_rasterize(pluto, pluto->clippath, path, &matrix, &pluto->clip, NULL, plutovg_fill_rule_non_zero);
plutovg_path_destroy(path);
}
plutovg_rle_t* rle = state->clippath ? state->clippath : pluto->clippath;
plutovg_blend(pluto, rle);
}
void plutovg_fill_preserve(plutovg_t* pluto)
{
plutovg_state_t* state = pluto->state;
plutovg_rle_clear(pluto->rle);
plutovg_rle_rasterize(pluto, pluto->rle, pluto->path, &state->matrix, &pluto->clip, NULL, state->winding);
plutovg_rle_clip_path(pluto->rle, state->clippath);
plutovg_blend(pluto, pluto->rle);
}
void plutovg_stroke_preserve(plutovg_t* pluto)
{
plutovg_state_t* state = pluto->state;
plutovg_rle_clear(pluto->rle);
plutovg_rle_rasterize(pluto, pluto->rle, pluto->path, &state->matrix, &pluto->clip, &state->stroke, plutovg_fill_rule_non_zero);
plutovg_rle_clip_path(pluto->rle, state->clippath);
plutovg_blend(pluto, pluto->rle);
}
void plutovg_clip_preserve(plutovg_t* pluto)
{
plutovg_state_t* state = pluto->state;
if(state->clippath)
{
plutovg_rle_clear(pluto->rle);
plutovg_rle_rasterize(pluto, pluto->rle, pluto->path, &state->matrix, &pluto->clip, NULL, state->winding);
plutovg_rle_clip_path(state->clippath, pluto->rle);
}
else
{
state->clippath = plutovg_rle_create();
plutovg_rle_rasterize(pluto, state->clippath, pluto->path, &state->matrix, &pluto->clip, NULL, state->winding);
}
}
void plutovg_reset_clip(plutovg_t* pluto)
{
plutovg_rle_destroy(pluto->state->clippath);
pluto->state->clippath = NULL;
}