#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, plutovg_texture_type_plain); 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; }