labwc/src/img/img-png.c

79 lines
1.9 KiB
C
Raw Normal View History

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (C) Johan Malm 2023
*/
#define _POSIX_C_SOURCE 200809L
#include "img/img-png.h"
#include <cairo.h>
#include <png.h>
#include <stdbool.h>
#include <stdio.h>
#include <wlr/util/log.h>
#include "buffer.h"
2024-01-19 19:06:07 +00:00
#include "common/string-helpers.h"
/*
* cairo_image_surface_create_from_png() does not gracefully handle non-png
* files, so we verify the header before trying to read the rest of the file.
*/
#define PNG_BYTES_TO_CHECK (4)
static bool
ispng(const char *filename)
{
unsigned char header[PNG_BYTES_TO_CHECK];
FILE *fp = fopen(filename, "rb");
if (!fp) {
return false;
}
if (fread(header, 1, PNG_BYTES_TO_CHECK, fp) != PNG_BYTES_TO_CHECK) {
fclose(fp);
return false;
}
if (png_sig_cmp(header, (png_size_t)0, PNG_BYTES_TO_CHECK)) {
wlr_log(WLR_ERROR, "file '%s' is not a recognised png file", filename);
fclose(fp);
return false;
}
fclose(fp);
return true;
}
#undef PNG_BYTES_TO_CHECK
struct lab_data_buffer *
img_png_load(const char *filename)
{
if (string_null_or_empty(filename)) {
return NULL;
}
if (!ispng(filename)) {
return NULL;
}
cairo_surface_t *image = cairo_image_surface_create_from_png(filename);
if (cairo_surface_status(image)) {
wlr_log(WLR_ERROR, "error reading png file '%s'", filename);
cairo_surface_destroy(image);
return NULL;
}
if (cairo_image_surface_get_format(image) == CAIRO_FORMAT_ARGB32) {
return buffer_adopt_cairo_surface(image);
} else {
/* Copy non-ARGB32 surface to ARGB32 buffer */
/* TODO: directly set non-ARGB32 surface in lab_data_buffer */
struct lab_data_buffer *buffer = buffer_create_cairo(
cairo_image_surface_get_width(image),
cairo_image_surface_get_height(image), 1);
cairo_t *cairo = cairo_create(buffer->surface);
cairo_set_source_surface(cairo, image, 0, 0);
cairo_paint(cairo);
cairo_surface_flush(cairo_get_target(cairo));
cairo_destroy(cairo);
cairo_surface_destroy(image);
return buffer;
}
}