/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- * * Copyright (C) 2014 Richard Hughes * * Licensed under the GNU Lesser General Public License Version 2.1 * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /** * SECTION:as-image * @short_description: Object representing a single image used in a screenshot. * @include: appstream-glib.h * @stability: Stable * * Screenshot may have multiple versions of an image in different resolutions * or aspect ratios. This object allows access to the location and size of a * single image. * * See also: #AsScreenshot */ #include "config.h" #include "as-cleanup.h" #include "as-image-private.h" #include "as-node-private.h" #include "as-utils-private.h" typedef struct _AsImagePrivate AsImagePrivate; struct _AsImagePrivate { AsImageKind kind; gchar *url; gchar *md5; gchar *basename; guint width; guint height; GdkPixbuf *pixbuf; }; G_DEFINE_TYPE_WITH_PRIVATE (AsImage, as_image, G_TYPE_OBJECT) #define GET_PRIVATE(o) (as_image_get_instance_private (o)) /** * as_image_finalize: **/ static void as_image_finalize (GObject *object) { AsImage *image = AS_IMAGE (object); AsImagePrivate *priv = GET_PRIVATE (image); if (priv->pixbuf != NULL) g_object_unref (priv->pixbuf); g_free (priv->url); g_free (priv->md5); g_free (priv->basename); G_OBJECT_CLASS (as_image_parent_class)->finalize (object); } /** * as_image_init: **/ static void as_image_init (AsImage *image) { } /** * as_image_class_init: **/ static void as_image_class_init (AsImageClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = as_image_finalize; } /** * as_image_kind_from_string: * @kind: the string. * * Converts the text representation to an enumerated value. * * Returns: (transfer full): a #AsImageKind, or %AS_IMAGE_KIND_UNKNOWN for unknown. * * Since: 0.1.0 **/ AsImageKind as_image_kind_from_string (const gchar *kind) { if (g_strcmp0 (kind, "source") == 0) return AS_IMAGE_KIND_SOURCE; if (g_strcmp0 (kind, "thumbnail") == 0) return AS_IMAGE_KIND_THUMBNAIL; return AS_IMAGE_KIND_UNKNOWN; } /** * as_image_kind_to_string: * @kind: the #AsImageKind. * * Converts the enumerated value to an text representation. * * Returns: string version of @kind * * Since: 0.1.0 **/ const gchar * as_image_kind_to_string (AsImageKind kind) { if (kind == AS_IMAGE_KIND_SOURCE) return "source"; if (kind == AS_IMAGE_KIND_THUMBNAIL) return "thumbnail"; return NULL; } /** * as_image_get_url: * @image: a #AsImage instance. * * Gets the full qualified URL for the image, usually pointing at some mirror. * * Returns: URL * * Since: 0.1.0 **/ const gchar * as_image_get_url (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->url; } /** * as_image_get_basename: * @image: a #AsImage instance. * * Gets the suggested basename the image, including file extension. * * Returns: filename * * Since: 0.1.6 **/ const gchar * as_image_get_basename (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->basename; } /** * as_image_get_md5: * @image: a #AsImage instance. * * Gets the string representation of the pixbuf hash value. * * Returns: string representing the MD5 sum, or %NULL if unset * * Since: 0.1.6 **/ const gchar * as_image_get_md5 (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->md5; } /** * as_image_get_width: * @image: a #AsImage instance. * * Gets the image width. * * Returns: width in pixels * * Since: 0.1.0 **/ guint as_image_get_width (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->width; } /** * as_image_get_height: * @image: a #AsImage instance. * * Gets the image height. * * Returns: height in pixels * * Since: 0.1.0 **/ guint as_image_get_height (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->height; } /** * as_image_get_kind: * @image: a #AsImage instance. * * Gets the image kind. * * Returns: the #AsImageKind * * Since: 0.1.0 **/ AsImageKind as_image_get_kind (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->kind; } /** * as_image_get_pixbuf: * @image: a #AsImage instance. * * Gets the image pixbuf if set. * * Returns: (transfer none): the #GdkPixbuf, or %NULL * * Since: 0.1.6 **/ GdkPixbuf * as_image_get_pixbuf (AsImage *image) { AsImagePrivate *priv = GET_PRIVATE (image); return priv->pixbuf; } /** * as_image_set_url: * @image: a #AsImage instance. * @url: the URL. * @url_len: the size of @url, or -1 if %NULL-terminated. * * Sets the fully-qualified mirror URL to use for the image. * * Since: 0.1.0 **/ void as_image_set_url (AsImage *image, const gchar *url, gssize url_len) { AsImagePrivate *priv = GET_PRIVATE (image); g_free (priv->url); priv->url = as_strndup (url, url_len); } /** * as_image_set_basename: * @image: a #AsImage instance. * @basename: the new filename basename. * * Sets the image basename filename. * * Since: 0.1.6 **/ void as_image_set_basename (AsImage *image, const gchar *basename) { AsImagePrivate *priv = GET_PRIVATE (image); g_free (priv->basename); priv->basename = g_strdup (basename); } /** * as_image_set_width: * @image: a #AsImage instance. * @width: the width in pixels. * * Sets the image width. * * Since: 0.1.0 **/ void as_image_set_width (AsImage *image, guint width) { AsImagePrivate *priv = GET_PRIVATE (image); priv->width = width; } /** * as_image_set_height: * @image: a #AsImage instance. * @height: the height in pixels. * * Sets the image height. * * Since: 0.1.0 **/ void as_image_set_height (AsImage *image, guint height) { AsImagePrivate *priv = GET_PRIVATE (image); priv->height = height; } /** * as_image_set_kind: * @image: a #AsImage instance. * @kind: the #AsImageKind, e.g. %AS_IMAGE_KIND_THUMBNAIL. * * Sets the image kind. * * Since: 0.1.0 **/ void as_image_set_kind (AsImage *image, AsImageKind kind) { AsImagePrivate *priv = GET_PRIVATE (image); priv->kind = kind; } /** * as_image_set_pixbuf: * @image: a #AsImage instance. * @pixbuf: the #GdkPixbuf, or %NULL * * Sets the image pixbuf. * * Since: 0.1.6 **/ void as_image_set_pixbuf (AsImage *image, GdkPixbuf *pixbuf) { AsImagePrivate *priv = GET_PRIVATE (image); guchar *data; guint len; if (priv->pixbuf != NULL) g_object_unref (priv->pixbuf); if (pixbuf == NULL) { priv->pixbuf = NULL; return; } if (priv->md5 == NULL) { data = gdk_pixbuf_get_pixels_with_length (pixbuf, &len); priv->md5 = g_compute_checksum_for_data (G_CHECKSUM_MD5, data, len); } priv->pixbuf = g_object_ref (pixbuf); priv->width = gdk_pixbuf_get_width (pixbuf); priv->height = gdk_pixbuf_get_height (pixbuf); } /** * as_image_node_insert: (skip) * @image: a #AsImage instance. * @parent: the parent #GNode to use.. * @api_version: the AppStream API version * * Inserts the image into the DOM tree. * * Returns: (transfer none): A populated #GNode * * Since: 0.1.0 **/ GNode * as_image_node_insert (AsImage *image, GNode *parent, gdouble api_version) { AsImagePrivate *priv = GET_PRIVATE (image); GNode *n; gchar tmp_height[6]; gchar tmp_width[6]; if (priv->width > 0 && priv->height > 0) { g_snprintf (tmp_width, sizeof (tmp_width), "%u", priv->width); g_snprintf (tmp_height, sizeof (tmp_height), "%u", priv->height); n = as_node_insert (parent, "image", priv->url, AS_NODE_INSERT_FLAG_NONE, "width", tmp_width, "height", tmp_height, "type", as_image_kind_to_string (priv->kind), NULL); } else { n = as_node_insert (parent, "image", priv->url, AS_NODE_INSERT_FLAG_NONE, "type", as_image_kind_to_string (priv->kind), NULL); } return n; } /** * as_image_node_parse: * @image: a #AsImage instance. * @node: a #GNode. * @error: A #GError or %NULL. * * Populates the object from a DOM node. * * Returns: %TRUE for success * * Since: 0.1.0 **/ gboolean as_image_node_parse (AsImage *image, GNode *node, GError **error) { AsImagePrivate *priv = GET_PRIVATE (image); const gchar *tmp; gchar *taken; guint size; size = as_node_get_attribute_as_int (node, "width"); if (size != G_MAXINT) as_image_set_width (image, size); size = as_node_get_attribute_as_int (node, "height"); if (size != G_MAXINT) as_image_set_height (image, size); tmp = as_node_get_attribute (node, "type"); if (tmp != NULL) as_image_set_kind (image, as_image_kind_from_string (tmp)); taken = as_node_take_data (node); if (taken != NULL) { g_free (priv->url); priv->url = taken; } return TRUE; } /** * as_image_load_filename: * @image: a #AsImage instance. * @filename: filename to read from * @error: A #GError or %NULL. * * Reads a pixbuf from a file. * * NOTE: This function also sets the suggested filename which can be retrieved * using as_image_get_basename(). This can be overridden if required. * * Returns: %TRUE for success * * Since: 0.1.6 **/ gboolean as_image_load_filename (AsImage *image, const gchar *filename, GError **error) { AsImagePrivate *priv = GET_PRIVATE (image); gsize len; _cleanup_free_ gchar *basename = NULL; _cleanup_free_ gchar *data = NULL; _cleanup_object_unref_ GdkPixbuf *pixbuf = NULL; /* get the contents so we can hash the predictable file data, * rather than the unpredicatable (for JPEG) pixel data */ if (!g_file_get_contents (filename, &data, &len, error)) return FALSE; g_free (priv->md5); priv->md5 = g_compute_checksum_for_data (G_CHECKSUM_MD5, (guchar * )data, len); /* load the image */ pixbuf = gdk_pixbuf_new_from_file (filename, error); if (pixbuf == NULL) return FALSE; /* set */ basename = g_path_get_basename (filename); as_image_set_basename (image, basename); as_image_set_pixbuf (image, pixbuf); return TRUE; } /** * as_image_save_pixbuf: * @image: a #AsImage instance. * @width: target width, or 0 for default * @height: target height, or 0 for default * @flags: some #AsImageSaveFlags values, e.g. %AS_IMAGE_SAVE_FLAG_PAD_16_9 * * Resamples a pixbuf to a specific size. * * Returns: (transfer full): A #GdkPixbuf of the specified size * * Since: 0.1.6 **/ GdkPixbuf * as_image_save_pixbuf (AsImage *image, guint width, guint height, AsImageSaveFlags flags) { AsImagePrivate *priv = GET_PRIVATE (image); GdkPixbuf *pixbuf = NULL; guint tmp_height; guint tmp_width; guint pixbuf_height; guint pixbuf_width; _cleanup_object_unref_ GdkPixbuf *pixbuf_tmp = NULL; /* 0 means 'default' */ if (width == 0) width = gdk_pixbuf_get_width (priv->pixbuf); if (height == 0) height = gdk_pixbuf_get_height (priv->pixbuf); /* don't do anything to an image with the correct size */ pixbuf_width = gdk_pixbuf_get_width (priv->pixbuf); pixbuf_height = gdk_pixbuf_get_height (priv->pixbuf); if (width == pixbuf_width && height == pixbuf_height) return g_object_ref (priv->pixbuf); /* never scale up, just pad */ if (pixbuf_width < width && pixbuf_height < height) { pixbuf = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8, width, height); gdk_pixbuf_fill (pixbuf, 0x00000000); gdk_pixbuf_copy_area (priv->pixbuf, 0, 0, /* of src */ pixbuf_width, pixbuf_height, pixbuf, (width - pixbuf_width) / 2, (height - pixbuf_height) / 2); return pixbuf; } /* is the aspect ratio of the source perfectly 16:9 */ if (flags == AS_IMAGE_SAVE_FLAG_NONE || (pixbuf_width / 16) * 9 == pixbuf_height) { pixbuf = gdk_pixbuf_scale_simple (priv->pixbuf, width, height, GDK_INTERP_HYPER); as_pixbuf_sharpen (pixbuf, 1, -0.5); return pixbuf; } /* create new 16:9 pixbuf with alpha padding */ pixbuf = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8, width, height); gdk_pixbuf_fill (pixbuf, 0x00000000); if ((pixbuf_width / 16) * 9 > pixbuf_height) { tmp_width = width; tmp_height = width * pixbuf_height / pixbuf_width; } else { tmp_width = height * pixbuf_width / pixbuf_height; tmp_height = height; } pixbuf_tmp = gdk_pixbuf_scale_simple (priv->pixbuf, tmp_width, tmp_height, GDK_INTERP_HYPER); as_pixbuf_sharpen (pixbuf_tmp, 1, -0.5); gdk_pixbuf_copy_area (pixbuf_tmp, 0, 0, /* of src */ tmp_width, tmp_height, pixbuf, (width - tmp_width) / 2, (height - tmp_height) / 2); return pixbuf; } /** * as_image_save_filename: * @image: a #AsImage instance. * @filename: filename to write to * @width: target width, or 0 for default * @height: target height, or 0 for default * @flags: some #AsImageSaveFlags values, e.g. %AS_IMAGE_SAVE_FLAG_PAD_16_9 * @error: A #GError or %NULL. * * Saves a pixbuf to a file. * * Returns: %TRUE for success * * Since: 0.1.6 **/ gboolean as_image_save_filename (AsImage *image, const gchar *filename, guint width, guint height, AsImageSaveFlags flags, GError **error) { _cleanup_object_unref_ GdkPixbuf *pixbuf; /* save source file */ pixbuf = as_image_save_pixbuf (image, width, height, flags); return gdk_pixbuf_save (pixbuf, filename, "png", error, NULL); } /** * is_pixel_alpha: **/ static gboolean is_pixel_alpha (GdkPixbuf *pixbuf, guint x, guint y) { gint rowstride, n_channels; guchar *pixels, *p; n_channels = gdk_pixbuf_get_n_channels (pixbuf); rowstride = gdk_pixbuf_get_rowstride (pixbuf); pixels = gdk_pixbuf_get_pixels (pixbuf); p = pixels + y * rowstride + x * n_channels; return p[3] == 0; } typedef enum { AS_IMAGE_ALPHA_MODE_START, AS_IMAGE_ALPHA_MODE_PADDING, AS_IMAGE_ALPHA_MODE_CONTENT, } AsImageAlphaMode; /** * as_image_get_alpha_flags: (skip) * @image: a #AsImage instance. * * Gets the alpha flags for the image. The following image would have all flags * set, where 'x' is alpha and '@' is non-alpha. * * xxxxxxxxxxxxxxxxxxxxxxxxxxxx * xx@@@@@@@@@@@@@@@@@@@@@@@@xx * xx@@@@@@@xxxxxx@@@@@@@@@@@xx * xx@@@@@@@xxxxxx@@@@@@@@@@@xx * xx@@@@@@@@@@@@@@@@@@@@@@@@xx * xxxxxxxxxxxxxxxxxxxxxxxxxxxx * * Returns: #AsImageAlphaFlags, e.g. %AS_IMAGE_ALPHA_FLAG_LEFT * * Since: 0.2.2 **/ AsImageAlphaFlags as_image_get_alpha_flags (AsImage *image) { AsImageAlphaFlags flags = AS_IMAGE_ALPHA_FLAG_TOP | AS_IMAGE_ALPHA_FLAG_BOTTOM | AS_IMAGE_ALPHA_FLAG_LEFT | AS_IMAGE_ALPHA_FLAG_RIGHT; AsImageAlphaMode mode_h; AsImageAlphaMode mode_v = AS_IMAGE_ALPHA_MODE_START; AsImagePrivate *priv = GET_PRIVATE (image); gboolean complete_line_of_alpha; gboolean is_alpha; gint width, height; gint x, y; guint cnt_content_to_alpha_h; guint cnt_content_to_alpha_v = 0; if (!gdk_pixbuf_get_has_alpha (priv->pixbuf)) return AS_IMAGE_ALPHA_FLAG_NONE; width = gdk_pixbuf_get_width (priv->pixbuf); height = gdk_pixbuf_get_height (priv->pixbuf); for (y = 0; y < height; y++) { mode_h = AS_IMAGE_ALPHA_MODE_START; complete_line_of_alpha = TRUE; cnt_content_to_alpha_h = 0; for (x = 0; x < width; x++) { is_alpha = is_pixel_alpha (priv->pixbuf, x, y); /* use the frame */ if (!is_alpha) { if (x == 0) flags &= ~AS_IMAGE_ALPHA_FLAG_LEFT; if (x == width - 1) flags &= ~AS_IMAGE_ALPHA_FLAG_RIGHT; if (y == 0) flags &= ~AS_IMAGE_ALPHA_FLAG_TOP; if (y == height - 1) flags &= ~AS_IMAGE_ALPHA_FLAG_BOTTOM; complete_line_of_alpha = FALSE; } /* use line state machine */ switch (mode_h) { case AS_IMAGE_ALPHA_MODE_START: mode_h = is_alpha ? AS_IMAGE_ALPHA_MODE_PADDING : AS_IMAGE_ALPHA_MODE_CONTENT; break; case AS_IMAGE_ALPHA_MODE_PADDING: if (!is_alpha) mode_h = AS_IMAGE_ALPHA_MODE_CONTENT; break; case AS_IMAGE_ALPHA_MODE_CONTENT: if (is_alpha) { mode_h = AS_IMAGE_ALPHA_MODE_PADDING; cnt_content_to_alpha_h++; } break; default: g_assert_not_reached (); } } /* use column state machine */ switch (mode_v) { case AS_IMAGE_ALPHA_MODE_START: if (complete_line_of_alpha) { mode_v = AS_IMAGE_ALPHA_MODE_PADDING; } else { mode_v = AS_IMAGE_ALPHA_MODE_CONTENT; } break; case AS_IMAGE_ALPHA_MODE_PADDING: if (!complete_line_of_alpha) mode_v = AS_IMAGE_ALPHA_MODE_CONTENT; break; case AS_IMAGE_ALPHA_MODE_CONTENT: if (complete_line_of_alpha) { mode_v = AS_IMAGE_ALPHA_MODE_PADDING; cnt_content_to_alpha_v++; } break; default: g_assert_not_reached (); } /* detect internal alpha */ if (mode_h == AS_IMAGE_ALPHA_MODE_PADDING) { if (cnt_content_to_alpha_h >= 2) flags |= AS_IMAGE_ALPHA_FLAG_INTERNAL; } else if (mode_h == AS_IMAGE_ALPHA_MODE_CONTENT) { if (cnt_content_to_alpha_h >= 1) flags |= AS_IMAGE_ALPHA_FLAG_INTERNAL; } } /* detect internal alpha */ if (mode_v == AS_IMAGE_ALPHA_MODE_PADDING) { if (cnt_content_to_alpha_v >= 2) flags |= AS_IMAGE_ALPHA_FLAG_INTERNAL; } else if (mode_v == AS_IMAGE_ALPHA_MODE_CONTENT) { if (cnt_content_to_alpha_v >= 1) flags |= AS_IMAGE_ALPHA_FLAG_INTERNAL; } return flags; } /** * as_image_new: * * Creates a new #AsImage. * * Returns: (transfer full): a #AsImage * * Since: 0.1.0 **/ AsImage * as_image_new (void) { AsImage *image; image = g_object_new (AS_TYPE_IMAGE, NULL); return AS_IMAGE (image); }