/* GIMP - The GNU Image Manipulation Program * 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 . */ #include "config.h" #include #include #include #include "libgimpbase/gimpbase.h" #include "core-types.h" #include "gegl/gimp-gegl-loops.h" #include "gimppattern.h" #include "gimppattern-load.h" #include "gimppattern-save.h" #include "gimptagged.h" #include "gimptempbuf.h" #include "gimp-intl.h" static void gimp_pattern_tagged_iface_init (GimpTaggedInterface *iface); static void gimp_pattern_finalize (GObject *object); static gint64 gimp_pattern_get_memsize (GimpObject *object, gint64 *gui_size); static gboolean gimp_pattern_get_size (GimpViewable *viewable, gint *width, gint *height); static GimpTempBuf * gimp_pattern_get_new_preview (GimpViewable *viewable, GimpContext *context, gint width, gint height); static gchar * gimp_pattern_get_description (GimpViewable *viewable, gchar **tooltip); static const gchar * gimp_pattern_get_extension (GimpData *data); static void gimp_pattern_copy (GimpData *data, GimpData *src_data); static gchar * gimp_pattern_get_checksum (GimpTagged *tagged); G_DEFINE_TYPE_WITH_CODE (GimpPattern, gimp_pattern, GIMP_TYPE_DATA, G_IMPLEMENT_INTERFACE (GIMP_TYPE_TAGGED, gimp_pattern_tagged_iface_init)) #define parent_class gimp_pattern_parent_class static void gimp_pattern_class_init (GimpPatternClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GimpObjectClass *gimp_object_class = GIMP_OBJECT_CLASS (klass); GimpViewableClass *viewable_class = GIMP_VIEWABLE_CLASS (klass); GimpDataClass *data_class = GIMP_DATA_CLASS (klass); object_class->finalize = gimp_pattern_finalize; gimp_object_class->get_memsize = gimp_pattern_get_memsize; viewable_class->default_icon_name = "gimp-tool-bucket-fill"; viewable_class->get_size = gimp_pattern_get_size; viewable_class->get_new_preview = gimp_pattern_get_new_preview; viewable_class->get_description = gimp_pattern_get_description; data_class->save = gimp_pattern_save; data_class->get_extension = gimp_pattern_get_extension; data_class->copy = gimp_pattern_copy; } static void gimp_pattern_tagged_iface_init (GimpTaggedInterface *iface) { iface->get_checksum = gimp_pattern_get_checksum; } static void gimp_pattern_init (GimpPattern *pattern) { pattern->mask = NULL; } static void gimp_pattern_finalize (GObject *object) { GimpPattern *pattern = GIMP_PATTERN (object); g_clear_pointer (&pattern->mask, gimp_temp_buf_unref); G_OBJECT_CLASS (parent_class)->finalize (object); } static gint64 gimp_pattern_get_memsize (GimpObject *object, gint64 *gui_size) { GimpPattern *pattern = GIMP_PATTERN (object); gint64 memsize = 0; memsize += gimp_temp_buf_get_memsize (pattern->mask); return memsize + GIMP_OBJECT_CLASS (parent_class)->get_memsize (object, gui_size); } static gboolean gimp_pattern_get_size (GimpViewable *viewable, gint *width, gint *height) { GimpPattern *pattern = GIMP_PATTERN (viewable); *width = gimp_temp_buf_get_width (pattern->mask); *height = gimp_temp_buf_get_height (pattern->mask); return TRUE; } static GimpTempBuf * gimp_pattern_get_new_preview (GimpViewable *viewable, GimpContext *context, gint width, gint height) { GimpPattern *pattern = GIMP_PATTERN (viewable); GimpTempBuf *temp_buf; GeglBuffer *src_buffer; gint true_width; gint true_height; gint copy_width; gint copy_height; gboolean has_temp_buf = FALSE; true_width = gimp_temp_buf_get_width (pattern->mask); true_height = gimp_temp_buf_get_height (pattern->mask); copy_width = MIN (width, true_width); copy_height = MIN (height, true_height); src_buffer = gimp_temp_buf_create_buffer (pattern->mask); if (true_width > width || true_height > height) { gdouble ratio_x = (gdouble) width / (gdouble) true_width; gdouble ratio_y = (gdouble) height / (gdouble) true_height; gdouble scale = MIN (ratio_x, ratio_y); gdouble aspect = (gdouble) true_width / (gdouble) true_height; /* Adjusting dimensions for non-square patterns */ if (true_width > true_height) copy_height = copy_width / aspect; else if (true_width < true_height) copy_width = copy_height * aspect; temp_buf = gimp_temp_buf_new (copy_width, copy_height, gimp_temp_buf_get_format (pattern->mask)); if (temp_buf) { gegl_buffer_get (src_buffer, GEGL_RECTANGLE (0, 0, copy_width, copy_height), scale, gimp_temp_buf_get_format (temp_buf), gimp_temp_buf_get_data (temp_buf), GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_CLAMP); has_temp_buf = TRUE; } } /* If scaled image pattern could not be loaded, * use the default pattern */ if (! has_temp_buf) { GeglBuffer *dest_buffer; temp_buf = gimp_temp_buf_new (copy_width, copy_height, gimp_temp_buf_get_format (pattern->mask)); dest_buffer = gimp_temp_buf_create_buffer (temp_buf); gimp_gegl_buffer_copy (src_buffer, GEGL_RECTANGLE (0, 0, copy_width, copy_height), GEGL_ABYSS_NONE, dest_buffer, GEGL_RECTANGLE (0, 0, 0, 0)); g_object_unref (dest_buffer); } g_object_unref (src_buffer); return temp_buf; } static gchar * gimp_pattern_get_description (GimpViewable *viewable, gchar **tooltip) { GimpPattern *pattern = GIMP_PATTERN (viewable); return g_strdup_printf ("%s (%d × %d)", gimp_object_get_name (pattern), gimp_temp_buf_get_width (pattern->mask), gimp_temp_buf_get_height (pattern->mask)); } static const gchar * gimp_pattern_get_extension (GimpData *data) { return GIMP_PATTERN_FILE_EXTENSION; } static void gimp_pattern_copy (GimpData *data, GimpData *src_data) { GimpPattern *pattern = GIMP_PATTERN (data); GimpPattern *src_pattern = GIMP_PATTERN (src_data); g_clear_pointer (&pattern->mask, gimp_temp_buf_unref); pattern->mask = gimp_temp_buf_copy (src_pattern->mask); gimp_data_dirty (data); } static gchar * gimp_pattern_get_checksum (GimpTagged *tagged) { GimpPattern *pattern = GIMP_PATTERN (tagged); gchar *checksum_string = NULL; if (pattern->mask) { GChecksum *checksum = g_checksum_new (G_CHECKSUM_MD5); g_checksum_update (checksum, gimp_temp_buf_get_data (pattern->mask), gimp_temp_buf_get_data_size (pattern->mask)); checksum_string = g_strdup (g_checksum_get_string (checksum)); g_checksum_free (checksum); } return checksum_string; } GimpData * gimp_pattern_new (GimpContext *context, const gchar *name) { GimpPattern *pattern; guchar *data; gint row, col; g_return_val_if_fail (name != NULL, NULL); g_return_val_if_fail (name[0] != '\n', NULL); pattern = g_object_new (GIMP_TYPE_PATTERN, "name", name, NULL); pattern->mask = gimp_temp_buf_new (32, 32, babl_format ("R'G'B' u8")); data = gimp_temp_buf_get_data (pattern->mask); for (row = 0; row < gimp_temp_buf_get_height (pattern->mask); row++) for (col = 0; col < gimp_temp_buf_get_width (pattern->mask); col++) { memset (data, (col % 2) && (row % 2) ? 255 : 0, 3); data += 3; } return GIMP_DATA (pattern); } GimpData * gimp_pattern_get_standard (GimpContext *context) { static GimpData *standard_pattern = NULL; if (! standard_pattern) { standard_pattern = gimp_pattern_new (context, "Standard"); gimp_data_clean (standard_pattern); gimp_data_make_internal (standard_pattern, "gimp-pattern-standard"); g_object_add_weak_pointer (G_OBJECT (standard_pattern), (gpointer *) &standard_pattern); } return standard_pattern; } GimpTempBuf * gimp_pattern_get_mask (GimpPattern *pattern) { g_return_val_if_fail (GIMP_IS_PATTERN (pattern), NULL); return pattern->mask; } GeglBuffer * gimp_pattern_create_buffer (GimpPattern *pattern) { g_return_val_if_fail (GIMP_IS_PATTERN (pattern), NULL); return gimp_temp_buf_create_buffer (pattern->mask); }