|
|
|
@ -323,7 +323,7 @@ struct attr { |
|
|
|
|
}; |
|
|
|
|
struct color_scheme { |
|
|
|
|
struct terminal_color palette[16]; |
|
|
|
|
struct terminal_color border; |
|
|
|
|
char border; |
|
|
|
|
struct attr default_attr; |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
@ -389,7 +389,7 @@ struct terminal { |
|
|
|
|
struct color_scheme *color_scheme; |
|
|
|
|
struct terminal_color color_table[256]; |
|
|
|
|
cairo_font_extents_t extents; |
|
|
|
|
cairo_font_face_t *font_normal, *font_bold; |
|
|
|
|
cairo_scaled_font_t *font_normal, *font_bold; |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
/* Create default tab stops, every 8 characters */ |
|
|
|
@ -473,7 +473,8 @@ terminal_get_row(struct terminal *terminal, int row) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static struct attr* |
|
|
|
|
terminal_get_attr_row(struct terminal *terminal, int row) { |
|
|
|
|
terminal_get_attr_row(struct terminal *terminal, int row) |
|
|
|
|
{ |
|
|
|
|
int index; |
|
|
|
|
|
|
|
|
|
index = (row + terminal->start) % terminal->height; |
|
|
|
@ -481,9 +482,54 @@ terminal_get_attr_row(struct terminal *terminal, int row) { |
|
|
|
|
return &terminal->data_attr[index * terminal->width]; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static struct attr |
|
|
|
|
terminal_get_attr(struct terminal *terminal, int row, int col) { |
|
|
|
|
return terminal_get_attr_row(terminal, row)[col]; |
|
|
|
|
union decoded_attr { |
|
|
|
|
struct { |
|
|
|
|
unsigned char foreground; |
|
|
|
|
unsigned char background; |
|
|
|
|
unsigned char bold; |
|
|
|
|
unsigned char underline; |
|
|
|
|
}; |
|
|
|
|
uint32_t key; |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
terminal_decode_attr(struct terminal *terminal, int row, int col, |
|
|
|
|
union decoded_attr *decoded) |
|
|
|
|
{ |
|
|
|
|
struct attr attr; |
|
|
|
|
int foreground, background, tmp; |
|
|
|
|
|
|
|
|
|
/* get the attributes for this character cell */ |
|
|
|
|
attr = terminal_get_attr_row(terminal, row)[col]; |
|
|
|
|
if ((attr.a & ATTRMASK_INVERSE) || |
|
|
|
|
((terminal->mode & MODE_SHOW_CURSOR) && |
|
|
|
|
terminal->focused && terminal->row == row && |
|
|
|
|
terminal->column == col)) { |
|
|
|
|
foreground = attr.bg; |
|
|
|
|
background = attr.fg; |
|
|
|
|
if (attr.a & ATTRMASK_BOLD) { |
|
|
|
|
if (foreground <= 16) foreground |= 0x08; |
|
|
|
|
if (background <= 16) background &= 0x07; |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
foreground = attr.fg; |
|
|
|
|
background = attr.bg; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (terminal->mode & MODE_INVERSE) { |
|
|
|
|
tmp = foreground; |
|
|
|
|
foreground = background; |
|
|
|
|
background = tmp; |
|
|
|
|
if (attr.a & ATTRMASK_BOLD) { |
|
|
|
|
if (foreground <= 16) foreground |= 0x08; |
|
|
|
|
if (background <= 16) background &= 0x07; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
decoded->foreground = foreground; |
|
|
|
|
decoded->background = background; |
|
|
|
|
decoded->bold = attr.a & (ATTRMASK_BOLD | ATTRMASK_BLINK); |
|
|
|
|
decoded->underline = attr.a & ATTRMASK_UNDERLINE; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
@ -610,12 +656,28 @@ terminal_resize(struct terminal *terminal, int width, int height) |
|
|
|
|
char *tab_ruler; |
|
|
|
|
int data_pitch, attr_pitch; |
|
|
|
|
int i, l, total_rows, start; |
|
|
|
|
struct rectangle rectangle; |
|
|
|
|
struct rectangle allocation; |
|
|
|
|
struct winsize ws; |
|
|
|
|
int32_t pixel_width, pixel_height; |
|
|
|
|
|
|
|
|
|
if (width < 1) |
|
|
|
|
width = 1; |
|
|
|
|
if (height < 1) |
|
|
|
|
height = 1; |
|
|
|
|
if (terminal->width == width && terminal->height == height) |
|
|
|
|
return; |
|
|
|
|
|
|
|
|
|
if (!terminal->fullscreen) { |
|
|
|
|
pixel_width = width * |
|
|
|
|
terminal->extents.max_x_advance + 2 * terminal->margin; |
|
|
|
|
pixel_height = height * |
|
|
|
|
terminal->extents.height + 2 * terminal->margin; |
|
|
|
|
window_set_child_size(terminal->window, |
|
|
|
|
pixel_width, pixel_height); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
window_schedule_redraw (terminal->window); |
|
|
|
|
|
|
|
|
|
data_pitch = width * sizeof(union utf8_char); |
|
|
|
|
size = data_pitch * height; |
|
|
|
|
data = malloc(size); |
|
|
|
@ -664,26 +726,12 @@ terminal_resize(struct terminal *terminal, int width, int height) |
|
|
|
|
terminal->tab_ruler = tab_ruler; |
|
|
|
|
terminal_init_tabs(terminal); |
|
|
|
|
|
|
|
|
|
if (terminal->row >= terminal->height) |
|
|
|
|
terminal->row = terminal->height - 1; |
|
|
|
|
if (terminal->column >= terminal->width) |
|
|
|
|
terminal->column = terminal->width - 1; |
|
|
|
|
terminal->start = 0; |
|
|
|
|
|
|
|
|
|
if (!terminal->fullscreen) { |
|
|
|
|
rectangle.width = terminal->width * |
|
|
|
|
terminal->extents.max_x_advance + 2 * terminal->margin; |
|
|
|
|
rectangle.height = terminal->height * |
|
|
|
|
terminal->extents.height + 2 * terminal->margin; |
|
|
|
|
window_set_child_size(terminal->window, &rectangle); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* Update the window size */ |
|
|
|
|
ws.ws_row = terminal->height; |
|
|
|
|
ws.ws_col = terminal->width; |
|
|
|
|
window_get_child_rectangle(terminal->window, &rectangle); |
|
|
|
|
ws.ws_xpixel = rectangle.width; |
|
|
|
|
ws.ws_ypixel = rectangle.height; |
|
|
|
|
window_get_child_allocation(terminal->window, &allocation); |
|
|
|
|
ws.ws_xpixel = allocation.width; |
|
|
|
|
ws.ws_ypixel = allocation.height; |
|
|
|
|
ioctl(terminal->master, TIOCSWINSZ, &ws); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -706,92 +754,125 @@ struct color_scheme DEFAULT_COLORS = { |
|
|
|
|
{0.33, 1, 1, 1}, /* high cyan */ |
|
|
|
|
{1, 1, 1, 1} /* white */ |
|
|
|
|
}, |
|
|
|
|
{0, 0, 0, 1}, /* black border */ |
|
|
|
|
0, /* black border */ |
|
|
|
|
{7, 0, 0, } /* bg:black (0), fg:light gray (7) */ |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
terminal_set_color(struct terminal *terminal, cairo_t *cr, int index) |
|
|
|
|
{ |
|
|
|
|
cairo_set_source_rgba(cr, |
|
|
|
|
terminal->color_table[index].r, |
|
|
|
|
terminal->color_table[index].g, |
|
|
|
|
terminal->color_table[index].b, |
|
|
|
|
terminal->color_table[index].a); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
struct glyph_run { |
|
|
|
|
struct terminal *terminal; |
|
|
|
|
cairo_t *cr; |
|
|
|
|
int count; |
|
|
|
|
union decoded_attr attr; |
|
|
|
|
cairo_glyph_t glyphs[256], *g; |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
glyph_run_init(struct glyph_run *run, struct terminal *terminal, cairo_t *cr) |
|
|
|
|
{ |
|
|
|
|
run->terminal = terminal; |
|
|
|
|
run->cr = cr; |
|
|
|
|
run->g = run->glyphs; |
|
|
|
|
run->count = 0; |
|
|
|
|
run->attr.key = 0; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
glyph_run_flush(struct glyph_run *run, union decoded_attr attr) |
|
|
|
|
{ |
|
|
|
|
cairo_scaled_font_t *font; |
|
|
|
|
|
|
|
|
|
if (run->count == 0) |
|
|
|
|
run->attr = attr; |
|
|
|
|
if (run->count > ARRAY_LENGTH(run->glyphs) - 10 || |
|
|
|
|
(attr.key != run->attr.key)) { |
|
|
|
|
if (run->attr.bold) |
|
|
|
|
font = run->terminal->font_bold; |
|
|
|
|
else |
|
|
|
|
font = run->terminal->font_normal; |
|
|
|
|
cairo_set_scaled_font(run->cr, font); |
|
|
|
|
terminal_set_color(run->terminal, run->cr, |
|
|
|
|
run->attr.foreground); |
|
|
|
|
|
|
|
|
|
cairo_show_glyphs (run->cr, run->glyphs, run->count); |
|
|
|
|
run->g = run->glyphs; |
|
|
|
|
run->count = 0; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
glyph_run_add(struct glyph_run *run, int x, int y, union utf8_char *c) |
|
|
|
|
{ |
|
|
|
|
int num_glyphs; |
|
|
|
|
cairo_scaled_font_t *font; |
|
|
|
|
|
|
|
|
|
num_glyphs = ARRAY_LENGTH(run->glyphs) - run->count; |
|
|
|
|
|
|
|
|
|
if (run->attr.bold) |
|
|
|
|
font = run->terminal->font_bold; |
|
|
|
|
else |
|
|
|
|
font = run->terminal->font_normal; |
|
|
|
|
|
|
|
|
|
cairo_move_to(run->cr, x, y); |
|
|
|
|
cairo_scaled_font_text_to_glyphs (font, x, y, |
|
|
|
|
(char *) c->byte, 4, |
|
|
|
|
&run->g, &num_glyphs, |
|
|
|
|
NULL, NULL, NULL); |
|
|
|
|
run->g += num_glyphs; |
|
|
|
|
run->count += num_glyphs; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
terminal_draw_contents(struct terminal *terminal) |
|
|
|
|
{ |
|
|
|
|
struct rectangle rectangle; |
|
|
|
|
struct rectangle allocation; |
|
|
|
|
cairo_t *cr; |
|
|
|
|
cairo_font_extents_t extents; |
|
|
|
|
int top_margin, side_margin; |
|
|
|
|
int row, col; |
|
|
|
|
struct attr attr; |
|
|
|
|
union utf8_char *p_row; |
|
|
|
|
struct utf8_chars { |
|
|
|
|
union utf8_char c; |
|
|
|
|
char null; |
|
|
|
|
} toShow; |
|
|
|
|
int foreground, background, bold, underline, concealed, tmp; |
|
|
|
|
union decoded_attr attr; |
|
|
|
|
int text_x, text_y; |
|
|
|
|
cairo_surface_t *surface; |
|
|
|
|
double d; |
|
|
|
|
struct glyph_run run; |
|
|
|
|
|
|
|
|
|
toShow.null = 0; |
|
|
|
|
|
|
|
|
|
window_get_child_rectangle(terminal->window, &rectangle); |
|
|
|
|
window_get_child_allocation(terminal->window, &allocation); |
|
|
|
|
|
|
|
|
|
surface = display_create_surface(terminal->display, &rectangle); |
|
|
|
|
surface = display_create_surface(terminal->display, &allocation); |
|
|
|
|
cr = cairo_create(surface); |
|
|
|
|
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE); |
|
|
|
|
cairo_set_source_rgba(cr, |
|
|
|
|
terminal->color_scheme->border.r, |
|
|
|
|
terminal->color_scheme->border.g, |
|
|
|
|
terminal->color_scheme->border.b, |
|
|
|
|
terminal->color_scheme->border.a); |
|
|
|
|
terminal_set_color(terminal, cr, terminal->color_scheme->border); |
|
|
|
|
cairo_paint(cr); |
|
|
|
|
cairo_set_operator(cr, CAIRO_OPERATOR_OVER); |
|
|
|
|
|
|
|
|
|
cairo_set_font_face(cr, terminal->font_normal); |
|
|
|
|
cairo_set_font_size(cr, 14); |
|
|
|
|
cairo_set_scaled_font(cr, terminal->font_normal); |
|
|
|
|
|
|
|
|
|
cairo_font_extents(cr, &extents); |
|
|
|
|
side_margin = (rectangle.width - terminal->width * extents.max_x_advance) / 2; |
|
|
|
|
top_margin = (rectangle.height - terminal->height * extents.height) / 2; |
|
|
|
|
side_margin = (allocation.width - terminal->width * extents.max_x_advance) / 2; |
|
|
|
|
top_margin = (allocation.height - terminal->height * extents.height) / 2; |
|
|
|
|
|
|
|
|
|
cairo_set_line_width(cr, 1.0); |
|
|
|
|
|
|
|
|
|
/* paint the background */ |
|
|
|
|
for (row = 0; row < terminal->height; row++) { |
|
|
|
|
p_row = terminal_get_row(terminal, row); |
|
|
|
|
for (col = 0; col < terminal->width; col++) { |
|
|
|
|
/* get the attributes for this character cell */ |
|
|
|
|
attr = terminal_get_attr(terminal, row, col); |
|
|
|
|
if ((attr.a & ATTRMASK_INVERSE) || |
|
|
|
|
((terminal->mode & MODE_SHOW_CURSOR) && |
|
|
|
|
terminal->focused && terminal->row == row && |
|
|
|
|
terminal->column == col)) |
|
|
|
|
{ |
|
|
|
|
foreground = attr.bg; |
|
|
|
|
background = attr.fg; |
|
|
|
|
if (attr.a & ATTRMASK_BOLD) { |
|
|
|
|
if (foreground <= 16) foreground |= 0x08; |
|
|
|
|
if (background <= 16) background &= 0x07; |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
foreground = attr.fg; |
|
|
|
|
background = attr.bg; |
|
|
|
|
} |
|
|
|
|
if (terminal->mode & MODE_INVERSE) { |
|
|
|
|
tmp = foreground; |
|
|
|
|
foreground = background; |
|
|
|
|
background = tmp; |
|
|
|
|
if (attr.a & ATTRMASK_BOLD) { |
|
|
|
|
if (foreground <= 16) foreground |= 0x08; |
|
|
|
|
if (background <= 16) background &= 0x07; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
bold = attr.a & (ATTRMASK_BOLD | ATTRMASK_BLINK); |
|
|
|
|
underline = attr.a & ATTRMASK_UNDERLINE; |
|
|
|
|
concealed = attr.a & ATTRMASK_CONCEALED; |
|
|
|
|
|
|
|
|
|
/* paint the background */ |
|
|
|
|
cairo_set_source_rgba(cr, |
|
|
|
|
terminal->color_table[background].r, |
|
|
|
|
terminal->color_table[background].g, |
|
|
|
|
terminal->color_table[background].b, |
|
|
|
|
terminal->color_table[background].a); |
|
|
|
|
terminal_decode_attr(terminal, row, col, &attr); |
|
|
|
|
|
|
|
|
|
if (attr.background == terminal->color_scheme->border) |
|
|
|
|
continue; |
|
|
|
|
|
|
|
|
|
terminal_set_color(terminal, cr, attr.background); |
|
|
|
|
cairo_move_to(cr, side_margin + (col * extents.max_x_advance), |
|
|
|
|
top_margin + (row * extents.height)); |
|
|
|
|
cairo_rel_line_to(cr, extents.max_x_advance, 0); |
|
|
|
@ -799,33 +880,36 @@ terminal_draw_contents(struct terminal *terminal) |
|
|
|
|
cairo_rel_line_to(cr, -extents.max_x_advance, 0); |
|
|
|
|
cairo_close_path(cr); |
|
|
|
|
cairo_fill(cr); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/* paint the foreground */ |
|
|
|
|
if (concealed) continue; |
|
|
|
|
if (bold) |
|
|
|
|
cairo_set_font_face(cr, terminal->font_bold); |
|
|
|
|
else |
|
|
|
|
cairo_set_font_face(cr, terminal->font_normal); |
|
|
|
|
cairo_set_source_rgba(cr, |
|
|
|
|
terminal->color_table[foreground].r, |
|
|
|
|
terminal->color_table[foreground].g, |
|
|
|
|
terminal->color_table[foreground].b, |
|
|
|
|
terminal->color_table[foreground].a); |
|
|
|
|
cairo_set_operator(cr, CAIRO_OPERATOR_OVER); |
|
|
|
|
|
|
|
|
|
/* paint the foreground */ |
|
|
|
|
glyph_run_init(&run, terminal, cr); |
|
|
|
|
for (row = 0; row < terminal->height; row++) { |
|
|
|
|
p_row = terminal_get_row(terminal, row); |
|
|
|
|
for (col = 0; col < terminal->width; col++) { |
|
|
|
|
/* get the attributes for this character cell */ |
|
|
|
|
terminal_decode_attr(terminal, row, col, &attr); |
|
|
|
|
|
|
|
|
|
glyph_run_flush(&run, attr); |
|
|
|
|
|
|
|
|
|
text_x = side_margin + col * extents.max_x_advance; |
|
|
|
|
text_y = top_margin + extents.ascent + row * extents.height; |
|
|
|
|
if (underline) { |
|
|
|
|
if (attr.underline) { |
|
|
|
|
cairo_move_to(cr, text_x, (double)text_y + 1.5); |
|
|
|
|
cairo_line_to(cr, text_x + extents.max_x_advance, (double) text_y + 1.5); |
|
|
|
|
cairo_stroke(cr); |
|
|
|
|
} |
|
|
|
|
cairo_move_to(cr, text_x, text_y); |
|
|
|
|
|
|
|
|
|
toShow.c = p_row[col]; |
|
|
|
|
cairo_show_text(cr, (char *) toShow.c.byte); |
|
|
|
|
|
|
|
|
|
glyph_run_add(&run, text_x, text_y, &p_row[col]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
attr.key = ~0; |
|
|
|
|
glyph_run_flush(&run, attr); |
|
|
|
|
|
|
|
|
|
if ((terminal->mode & MODE_SHOW_CURSOR) && !terminal->focused) { |
|
|
|
|
d = 0.5; |
|
|
|
|
|
|
|
|
@ -842,31 +926,29 @@ terminal_draw_contents(struct terminal *terminal) |
|
|
|
|
|
|
|
|
|
cairo_destroy(cr); |
|
|
|
|
|
|
|
|
|
window_copy_surface(terminal->window, |
|
|
|
|
&rectangle, |
|
|
|
|
surface); |
|
|
|
|
window_copy_surface(terminal->window, &allocation, surface); |
|
|
|
|
|
|
|
|
|
cairo_surface_destroy(surface); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
terminal_draw(struct terminal *terminal) |
|
|
|
|
resize_handler(struct window *window, |
|
|
|
|
int32_t pixel_width, int32_t pixel_height, void *data) |
|
|
|
|
{ |
|
|
|
|
struct rectangle rectangle; |
|
|
|
|
struct terminal *terminal = data; |
|
|
|
|
int32_t width, height; |
|
|
|
|
|
|
|
|
|
window_get_child_rectangle(terminal->window, &rectangle); |
|
|
|
|
|
|
|
|
|
width = (rectangle.width - 2 * terminal->margin) / |
|
|
|
|
width = (pixel_width - 2 * terminal->margin) / |
|
|
|
|
(int32_t) terminal->extents.max_x_advance; |
|
|
|
|
height = (rectangle.height - 2 * terminal->margin) / |
|
|
|
|
height = (pixel_height - 2 * terminal->margin) / |
|
|
|
|
(int32_t) terminal->extents.height; |
|
|
|
|
|
|
|
|
|
if (width < 0 || height < 0) |
|
|
|
|
return; |
|
|
|
|
|
|
|
|
|
terminal_resize(terminal, width, height); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void |
|
|
|
|
terminal_draw(struct terminal *terminal) |
|
|
|
|
{ |
|
|
|
|
window_draw(terminal->window); |
|
|
|
|
terminal_draw_contents(terminal); |
|
|
|
|
window_flush(terminal->window); |
|
|
|
@ -1960,6 +2042,7 @@ terminal_create(struct display *display, int fullscreen) |
|
|
|
|
window_set_fullscreen(terminal->window, terminal->fullscreen); |
|
|
|
|
window_set_user_data(terminal->window, terminal); |
|
|
|
|
window_set_redraw_handler(terminal->window, redraw_handler); |
|
|
|
|
window_set_resize_handler(terminal->window, resize_handler); |
|
|
|
|
|
|
|
|
|
window_set_key_handler(terminal->window, key_handler); |
|
|
|
|
window_set_keyboard_focus_handler(terminal->window, |
|
|
|
@ -1967,16 +2050,19 @@ terminal_create(struct display *display, int fullscreen) |
|
|
|
|
|
|
|
|
|
surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 0, 0); |
|
|
|
|
cr = cairo_create(surface); |
|
|
|
|
terminal->font_bold = cairo_toy_font_face_create ("mono", |
|
|
|
|
CAIRO_FONT_SLANT_NORMAL, |
|
|
|
|
CAIRO_FONT_WEIGHT_BOLD); |
|
|
|
|
cairo_font_face_reference(terminal->font_bold); |
|
|
|
|
terminal->font_normal = cairo_toy_font_face_create ("mono", |
|
|
|
|
CAIRO_FONT_SLANT_NORMAL, |
|
|
|
|
CAIRO_FONT_WEIGHT_NORMAL); |
|
|
|
|
cairo_font_face_reference(terminal->font_normal); |
|
|
|
|
cairo_set_font_face(cr, terminal->font_normal); |
|
|
|
|
cairo_set_font_size(cr, 14); |
|
|
|
|
cairo_select_font_face (cr, "mono", |
|
|
|
|
CAIRO_FONT_SLANT_NORMAL, |
|
|
|
|
CAIRO_FONT_WEIGHT_BOLD); |
|
|
|
|
terminal->font_bold = cairo_get_scaled_font (cr); |
|
|
|
|
cairo_scaled_font_reference(terminal->font_bold); |
|
|
|
|
|
|
|
|
|
cairo_select_font_face (cr, "mono", |
|
|
|
|
CAIRO_FONT_SLANT_NORMAL, |
|
|
|
|
CAIRO_FONT_WEIGHT_NORMAL); |
|
|
|
|
terminal->font_normal = cairo_get_scaled_font (cr); |
|
|
|
|
cairo_scaled_font_reference(terminal->font_normal); |
|
|
|
|
|
|
|
|
|
cairo_font_extents(cr, &terminal->extents); |
|
|
|
|
cairo_destroy(cr); |
|
|
|
|
cairo_surface_destroy(surface); |
|
|
|
|