mirror of
https://github.com/RetroDECK/ES-DE.git
synced 2025-01-18 15:15:37 +00:00
496 lines
13 KiB
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;
|
|
}
|