PIKApp/app/core/pikapattern-load.c

225 lines
7.1 KiB
C
Raw Normal View History

2023-09-26 00:35:21 +02:00
/* PIKA - Photo and Image Kooker Application
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
*
* Original copyright, applying to most contents (license remains unchanged):
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <cairo.h>
#include <gegl.h>
#include <gdk-pixbuf/gdk-pixbuf.h>
#include "libpikabase/pikabase.h"
#include "libpikacolor/pikacolor.h"
#include "core-types.h"
#include "pikapattern.h"
#include "pikapattern-header.h"
#include "pikapattern-load.h"
#include "pikatempbuf.h"
#include "pika-intl.h"
GList *
pika_pattern_load (PikaContext *context,
GFile *file,
GInputStream *input,
GError **error)
{
PikaPattern *pattern = NULL;
const Babl *format = NULL;
PikaPatternHeader header;
gsize size;
gsize bytes_read;
gsize bn_size;
gchar *name = NULL;
g_return_val_if_fail (G_IS_FILE (file), NULL);
g_return_val_if_fail (G_IS_INPUT_STREAM (input), NULL);
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
/* read the size */
if (! g_input_stream_read_all (input, &header, sizeof (header),
&bytes_read, NULL, error) ||
bytes_read != sizeof (header))
{
g_prefix_error (error, _("File appears truncated: "));
goto error;
}
/* rearrange the bytes in each unsigned int */
header.header_size = g_ntohl (header.header_size);
header.version = g_ntohl (header.version);
header.width = g_ntohl (header.width);
header.height = g_ntohl (header.height);
header.bytes = g_ntohl (header.bytes);
header.magic_number = g_ntohl (header.magic_number);
/* Check for correct file format */
if (header.magic_number != PIKA_PATTERN_MAGIC ||
header.version != 1 ||
header.header_size <= sizeof (header))
{
g_set_error (error, PIKA_DATA_ERROR, PIKA_DATA_ERROR_READ,
_("Unknown pattern format version %d."),
header.version);
goto error;
}
/* Check for supported bit depths */
if (header.bytes < 1 || header.bytes > 4)
{
g_set_error (error, PIKA_DATA_ERROR, PIKA_DATA_ERROR_READ,
_("Unsupported pattern depth %d.\n"
"PIKA Patterns must be GRAY or RGB."),
header.bytes);
goto error;
}
/* Validate dimensions */
if ((header.width == 0) || (header.width > PIKA_PATTERN_MAX_SIZE) ||
(header.height == 0) || (header.height > PIKA_PATTERN_MAX_SIZE) ||
(G_MAXSIZE / header.width / header.height / header.bytes < 1))
{
g_set_error (error, PIKA_DATA_ERROR, PIKA_DATA_ERROR_READ,
_("Invalid header data in '%s': width=%lu (maximum %lu), "
"height=%lu (maximum %lu), bytes=%lu"),
pika_file_get_utf8_name (file),
(gulong) header.width, (gulong) PIKA_PATTERN_MAX_SIZE,
(gulong) header.height, (gulong) PIKA_PATTERN_MAX_SIZE,
(gulong) header.bytes);
goto error;
}
/* Read in the pattern name */
if ((bn_size = (header.header_size - sizeof (header))))
{
gchar *utf8;
if (bn_size > PIKA_PATTERN_MAX_NAME)
{
g_set_error (error, PIKA_DATA_ERROR, PIKA_DATA_ERROR_READ,
_("Invalid header data in '%s': "
"Pattern name is too long: %lu"),
pika_file_get_utf8_name (file),
(gulong) bn_size);
goto error;
}
name = g_new0 (gchar, bn_size + 1);
if (! g_input_stream_read_all (input, name, bn_size,
&bytes_read, NULL, error) ||
bytes_read != bn_size)
{
g_prefix_error (error, _("File appears truncated."));
g_free (name);
goto error;
}
utf8 = pika_any_to_utf8 (name, bn_size - 1,
_("Invalid UTF-8 string in pattern file '%s'."),
pika_file_get_utf8_name (file));
g_free (name);
name = utf8;
}
if (! name)
name = g_strdup (_("Unnamed"));
pattern = g_object_new (PIKA_TYPE_PATTERN,
"name", name,
"mime-type", "image/x-pika-pat",
NULL);
g_free (name);
switch (header.bytes)
{
case 1: format = babl_format ("Y' u8"); break;
case 2: format = babl_format ("Y'A u8"); break;
case 3: format = babl_format ("R'G'B' u8"); break;
case 4: format = babl_format ("R'G'B'A u8"); break;
}
pattern->mask = pika_temp_buf_new (header.width, header.height, format);
size = (gsize) header.width * header.height * header.bytes;
if (! g_input_stream_read_all (input,
pika_temp_buf_get_data (pattern->mask), size,
&bytes_read, NULL, error) ||
bytes_read != size)
{
g_prefix_error (error, _("File appears truncated."));
goto error;
}
return g_list_prepend (NULL, pattern);
error:
if (pattern)
g_object_unref (pattern);
g_prefix_error (error, _("Fatal parse error in pattern file: "));
return NULL;
}
GList *
pika_pattern_load_pixbuf (PikaContext *context,
GFile *file,
GInputStream *input,
GError **error)
{
PikaPattern *pattern;
GdkPixbuf *pixbuf;
gchar *name;
g_return_val_if_fail (G_IS_FILE (file), NULL);
g_return_val_if_fail (G_IS_INPUT_STREAM (input), NULL);
g_return_val_if_fail (error == NULL || *error == NULL, NULL);
pixbuf = gdk_pixbuf_new_from_stream (input, NULL, error);
if (! pixbuf)
return NULL;
name = g_strdup (gdk_pixbuf_get_option (pixbuf, "tEXt::Title"));
if (! name)
name = g_strdup (gdk_pixbuf_get_option (pixbuf, "tEXt::Comment"));
if (! name)
name = g_path_get_basename (pika_file_get_utf8_name (file));
pattern = g_object_new (PIKA_TYPE_PATTERN,
"name", name,
"mime-type", NULL, /* FIXME!! */
NULL);
g_free (name);
pattern->mask = pika_temp_buf_new_from_pixbuf (pixbuf, NULL);
g_object_unref (pixbuf);
return g_list_prepend (NULL, pattern);
}