220 lines
6.7 KiB
C
220 lines
6.7 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- */
|
|
/* GdkPixbuf library - SVG image loader
|
|
*
|
|
* Copyright (C) 2002 Matthias Clasen
|
|
* Copyright (C) 2002-2004 Dom Lachowicz
|
|
*
|
|
* Authors: Matthias Clasen <maclas@gmx.de>
|
|
* Dom Lachowicz <cinamod@hotmail.com>
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#define RSVG_DISABLE_DEPRECATION_WARNINGS
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <librsvg/rsvg.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
|
|
typedef struct {
|
|
RsvgHandle *handle;
|
|
|
|
GdkPixbufModuleUpdatedFunc updated_func;
|
|
GdkPixbufModulePreparedFunc prepared_func;
|
|
GdkPixbufModuleSizeFunc size_func;
|
|
|
|
gboolean first_write;
|
|
|
|
gpointer user_data;
|
|
} SvgContext;
|
|
|
|
G_MODULE_EXPORT void fill_vtable (GdkPixbufModule *module);
|
|
G_MODULE_EXPORT void fill_info (GdkPixbufFormat *info);
|
|
|
|
enum {
|
|
ERROR_WRITING = 1,
|
|
ERROR_DISPLAYING_IMAGE
|
|
} RsvgLoaderErrorReasons;
|
|
|
|
static void
|
|
rsvg_propagate_error (GError ** err,
|
|
const char * reason,
|
|
gint code)
|
|
{
|
|
if (err) {
|
|
*err = NULL;
|
|
g_set_error (err, rsvg_error_quark (), code, "%s", reason);
|
|
}
|
|
}
|
|
|
|
static gpointer
|
|
gdk_pixbuf__svg_image_begin_load (GdkPixbufModuleSizeFunc size_func,
|
|
GdkPixbufModulePreparedFunc prepared_func,
|
|
GdkPixbufModuleUpdatedFunc updated_func,
|
|
gpointer user_data,
|
|
GError **error)
|
|
{
|
|
SvgContext *context = g_new0 (SvgContext, 1);
|
|
|
|
if (error)
|
|
*error = NULL;
|
|
|
|
context->first_write = TRUE;
|
|
context->size_func = size_func;
|
|
|
|
context->prepared_func = prepared_func;
|
|
context->updated_func = updated_func;
|
|
context->user_data = user_data;
|
|
|
|
return context;
|
|
}
|
|
|
|
static void
|
|
emit_updated (SvgContext *context, GdkPixbuf *pixbuf)
|
|
{
|
|
if (context->updated_func != NULL)
|
|
(* context->updated_func) (pixbuf,
|
|
0, 0,
|
|
gdk_pixbuf_get_width (pixbuf),
|
|
gdk_pixbuf_get_height (pixbuf),
|
|
context->user_data);
|
|
}
|
|
|
|
static void
|
|
emit_prepared (SvgContext *context, GdkPixbuf *pixbuf)
|
|
{
|
|
if (context->prepared_func != NULL)
|
|
(* context->prepared_func) (pixbuf, NULL, context->user_data);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pixbuf__svg_image_load_increment (gpointer data,
|
|
const guchar *buf, guint size,
|
|
GError **error)
|
|
{
|
|
SvgContext *context = (SvgContext *)data;
|
|
|
|
if (error)
|
|
*error = NULL;
|
|
|
|
if (context->first_write == TRUE) {
|
|
context->first_write = FALSE;
|
|
|
|
context->handle = rsvg_handle_new ();
|
|
|
|
if (!context->handle) {
|
|
rsvg_propagate_error (error, "Error displaying image", ERROR_DISPLAYING_IMAGE);
|
|
return FALSE;
|
|
}
|
|
|
|
rsvg_handle_set_size_callback (context->handle, context->size_func, context->user_data, NULL);
|
|
}
|
|
|
|
if (!context->handle) {
|
|
rsvg_propagate_error (error, "Error displaying image", ERROR_DISPLAYING_IMAGE);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!rsvg_handle_write (context->handle, buf, size, error)) {
|
|
g_clear_error(error);
|
|
rsvg_propagate_error (error, "Error writing", ERROR_WRITING);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pixbuf__svg_image_stop_load (gpointer data, GError **error)
|
|
{
|
|
SvgContext *context = (SvgContext *)data;
|
|
GdkPixbuf *pixbuf;
|
|
gboolean result = TRUE;
|
|
|
|
if (error)
|
|
*error = NULL;
|
|
|
|
if (!context->handle) {
|
|
rsvg_propagate_error (error, "Error displaying image", ERROR_DISPLAYING_IMAGE);
|
|
return FALSE;
|
|
}
|
|
|
|
if (!rsvg_handle_close (context->handle, error)) {
|
|
g_object_unref (context->handle);
|
|
g_free (context);
|
|
return FALSE;
|
|
}
|
|
|
|
pixbuf = rsvg_handle_get_pixbuf (context->handle);
|
|
|
|
if (pixbuf != NULL) {
|
|
emit_prepared (context, pixbuf);
|
|
emit_updated (context, pixbuf);
|
|
g_object_unref (pixbuf);
|
|
}
|
|
else {
|
|
rsvg_propagate_error (error, "Error displaying image", ERROR_DISPLAYING_IMAGE);
|
|
result = FALSE;
|
|
}
|
|
|
|
g_object_unref (context->handle);
|
|
g_free (context);
|
|
|
|
return result;
|
|
}
|
|
|
|
void
|
|
fill_vtable (GdkPixbufModule *module)
|
|
{
|
|
module->begin_load = gdk_pixbuf__svg_image_begin_load;
|
|
module->stop_load = gdk_pixbuf__svg_image_stop_load;
|
|
module->load_increment = gdk_pixbuf__svg_image_load_increment;
|
|
}
|
|
|
|
void
|
|
fill_info (GdkPixbufFormat *info)
|
|
{
|
|
static const GdkPixbufModulePattern signature[] = {
|
|
{ " <svg", "* ", 100 },
|
|
{ " <!DOCTYPE svg", "* ", 100 },
|
|
{ NULL, NULL, 0 }
|
|
};
|
|
|
|
static const gchar *mime_types[] = { /* yes folks, i actually have run into all of these in the wild... */
|
|
"image/svg+xml",
|
|
"image/svg",
|
|
"image/svg-xml",
|
|
"image/vnd.adobe.svg+xml",
|
|
"text/xml-svg",
|
|
"image/svg+xml-compressed",
|
|
NULL
|
|
};
|
|
static const gchar *extensions[] = {
|
|
"svg",
|
|
"svgz",
|
|
"svg.gz",
|
|
NULL
|
|
};
|
|
|
|
info->name = "svg";
|
|
info->signature = (GdkPixbufModulePattern *) signature;
|
|
info->description = "Scalable Vector Graphics";
|
|
info->mime_types = (gchar **) mime_types;
|
|
info->extensions = (gchar **) extensions;
|
|
info->flags = GDK_PIXBUF_FORMAT_SCALABLE | GDK_PIXBUF_FORMAT_THREADSAFE;
|
|
info->license = "LGPL";
|
|
}
|