1017 lines
36 KiB
C
1017 lines
36 KiB
C
/* 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
|
|
*
|
|
* PikaText
|
|
* Copyright (C) 2002-2003 Sven Neumann <sven@gimp.org>
|
|
*
|
|
* 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 <string.h>
|
|
|
|
#include <cairo.h>
|
|
#include <gegl.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <pango/pango.h>
|
|
|
|
#include "libpikabase/pikabase.h"
|
|
#include "libpikamath/pikamath.h"
|
|
#include "libpikacolor/pikacolor.h"
|
|
#include "libpikaconfig/pikaconfig.h"
|
|
|
|
#include "text-types.h"
|
|
|
|
#include "pikafont.h"
|
|
|
|
#include "core/pika.h"
|
|
#include "core/pika-memsize.h"
|
|
#include "core/pika-utils.h"
|
|
#include "core/pikacontainer.h"
|
|
#include "core/pikadashpattern.h"
|
|
#include "core/pikadatafactory.h"
|
|
#include "core/pikastrokeoptions.h"
|
|
#include "core/pikapattern.h"
|
|
|
|
#include "pikatext.h"
|
|
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_PIKA,
|
|
PROP_TEXT,
|
|
PROP_MARKUP,
|
|
PROP_FONT,
|
|
PROP_FONT_SIZE,
|
|
PROP_UNIT,
|
|
PROP_ANTIALIAS,
|
|
PROP_HINT_STYLE,
|
|
PROP_KERNING,
|
|
PROP_LANGUAGE,
|
|
PROP_BASE_DIR,
|
|
PROP_COLOR,
|
|
PROP_OUTLINE,
|
|
PROP_JUSTIFICATION,
|
|
PROP_INDENTATION,
|
|
PROP_LINE_SPACING,
|
|
PROP_LETTER_SPACING,
|
|
PROP_BOX_MODE,
|
|
PROP_BOX_WIDTH,
|
|
PROP_BOX_HEIGHT,
|
|
PROP_BOX_UNIT,
|
|
PROP_TRANSFORMATION,
|
|
PROP_OFFSET_X,
|
|
PROP_OFFSET_Y,
|
|
PROP_BORDER,
|
|
|
|
PROP_OUTLINE_STYLE, /* fill-options */
|
|
PROP_OUTLINE_FOREGROUND, /* context */
|
|
PROP_OUTLINE_PATTERN, /* context */
|
|
PROP_OUTLINE_WIDTH, /* stroke-options */
|
|
PROP_OUTLINE_UNIT,
|
|
PROP_OUTLINE_CAP_STYLE,
|
|
PROP_OUTLINE_JOIN_STYLE,
|
|
PROP_OUTLINE_MITER_LIMIT,
|
|
PROP_OUTLINE_ANTIALIAS, /* fill-options */
|
|
PROP_OUTLINE_DASH_OFFSET,
|
|
PROP_OUTLINE_DASH_INFO,
|
|
/* for backward compatibility */
|
|
PROP_HINTING
|
|
};
|
|
|
|
enum
|
|
{
|
|
CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static void pika_text_config_iface_init (PikaConfigInterface *iface);
|
|
static gboolean pika_text_serialize_property (PikaConfig *config,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec,
|
|
PikaConfigWriter *writer);
|
|
static gboolean pika_text_deserialize_property (PikaConfig *config,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec,
|
|
GScanner *scanner,
|
|
GTokenType *expected);
|
|
|
|
static void pika_text_finalize (GObject *object);
|
|
static void pika_text_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void pika_text_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void pika_text_dispatch_properties_changed (GObject *object,
|
|
guint n_pspecs,
|
|
GParamSpec **pspecs);
|
|
static gint64 pika_text_get_memsize (PikaObject *object,
|
|
gint64 *gui_size);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (PikaText, pika_text, PIKA_TYPE_OBJECT,
|
|
G_IMPLEMENT_INTERFACE (PIKA_TYPE_CONFIG,
|
|
pika_text_config_iface_init))
|
|
|
|
#define parent_class pika_text_parent_class
|
|
|
|
static guint text_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
static void
|
|
pika_text_class_init (PikaTextClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
PikaObjectClass *pika_object_class = PIKA_OBJECT_CLASS (klass);
|
|
PikaRGB black;
|
|
PikaRGB gray;
|
|
PikaMatrix2 identity;
|
|
gchar *language;
|
|
GParamSpec *array_spec;
|
|
|
|
text_signals[CHANGED] =
|
|
g_signal_new ("changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (PikaTextClass, changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
object_class->finalize = pika_text_finalize;
|
|
object_class->get_property = pika_text_get_property;
|
|
object_class->set_property = pika_text_set_property;
|
|
object_class->dispatch_properties_changed = pika_text_dispatch_properties_changed;
|
|
|
|
pika_object_class->get_memsize = pika_text_get_memsize;
|
|
|
|
pika_rgba_set (&black, 0.0, 0.0, 0.0, PIKA_OPACITY_OPAQUE);
|
|
pika_rgba_set (&gray, 0.75, 0.75, 0.75, PIKA_OPACITY_OPAQUE);
|
|
pika_matrix2_identity (&identity);
|
|
|
|
PIKA_CONFIG_PROP_STRING (object_class, PROP_TEXT,
|
|
"text",
|
|
NULL, NULL,
|
|
NULL,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_STRING (object_class, PROP_MARKUP,
|
|
"markup",
|
|
NULL, NULL,
|
|
NULL,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_FONT (object_class, PROP_FONT,
|
|
"font", NULL, NULL,
|
|
PIKA_CONFIG_PARAM_FLAGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_FONT_SIZE,
|
|
"font-size",
|
|
NULL, NULL,
|
|
0.0, 8192.0, 24.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
/* We use the name "font-size-unit" for backward compatibility.
|
|
* The unit is also used for other sizes in the text object.
|
|
*/
|
|
PIKA_CONFIG_PROP_UNIT (object_class, PROP_UNIT,
|
|
"font-size-unit",
|
|
NULL, NULL,
|
|
TRUE, FALSE, PIKA_UNIT_PIXEL,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_ANTIALIAS,
|
|
"antialias",
|
|
NULL, NULL,
|
|
TRUE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_HINT_STYLE,
|
|
"hint-style",
|
|
NULL, NULL,
|
|
PIKA_TYPE_TEXT_HINT_STYLE,
|
|
PIKA_TEXT_HINT_STYLE_MEDIUM,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_KERNING,
|
|
"kerning",
|
|
NULL, NULL,
|
|
FALSE,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
language = pika_get_default_language (NULL);
|
|
|
|
PIKA_CONFIG_PROP_STRING (object_class, PROP_LANGUAGE,
|
|
"language",
|
|
NULL, NULL,
|
|
language,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
g_free (language);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_BASE_DIR,
|
|
"base-direction",
|
|
NULL, NULL,
|
|
PIKA_TYPE_TEXT_DIRECTION,
|
|
PIKA_TEXT_DIRECTION_LTR,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_RGB (object_class, PROP_COLOR,
|
|
"color",
|
|
NULL, NULL,
|
|
FALSE, &black,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_OUTLINE,
|
|
"outline",
|
|
NULL, NULL,
|
|
PIKA_TYPE_TEXT_OUTLINE,
|
|
PIKA_TEXT_OUTLINE_NONE,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_JUSTIFICATION,
|
|
"justify",
|
|
NULL, NULL,
|
|
PIKA_TYPE_TEXT_JUSTIFICATION,
|
|
PIKA_TEXT_JUSTIFY_LEFT,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_INDENTATION,
|
|
"indent",
|
|
NULL, NULL,
|
|
-8192.0, 8192.0, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_LINE_SPACING,
|
|
"line-spacing",
|
|
NULL, NULL,
|
|
-8192.0, 8192.0, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_LETTER_SPACING,
|
|
"letter-spacing",
|
|
NULL, NULL,
|
|
-8192.0, 8192.0, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_BOX_MODE,
|
|
"box-mode",
|
|
NULL, NULL,
|
|
PIKA_TYPE_TEXT_BOX_MODE,
|
|
PIKA_TEXT_BOX_DYNAMIC,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_BOX_WIDTH,
|
|
"box-width",
|
|
NULL, NULL,
|
|
0.0, PIKA_MAX_IMAGE_SIZE, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_BOX_HEIGHT,
|
|
"box-height",
|
|
NULL, NULL,
|
|
0.0, PIKA_MAX_IMAGE_SIZE, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_UNIT (object_class, PROP_BOX_UNIT,
|
|
"box-unit",
|
|
NULL, NULL,
|
|
TRUE, FALSE, PIKA_UNIT_PIXEL,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
PIKA_CONFIG_PROP_MATRIX2 (object_class, PROP_TRANSFORMATION,
|
|
"transformation",
|
|
NULL, NULL,
|
|
&identity,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_OFFSET_X,
|
|
"offset-x",
|
|
NULL, NULL,
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_OFFSET_Y,
|
|
"offset-y",
|
|
NULL, NULL,
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
|
|
/* border does only exist to implement the old text API */
|
|
g_object_class_install_property (object_class, PROP_BORDER,
|
|
g_param_spec_int ("border", NULL, NULL,
|
|
0, PIKA_MAX_IMAGE_SIZE, 0,
|
|
G_PARAM_CONSTRUCT |
|
|
PIKA_PARAM_WRITABLE));
|
|
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_OUTLINE_STYLE,
|
|
"outline-custom-style", NULL, NULL,
|
|
PIKA_TYPE_CUSTOM_STYLE,
|
|
PIKA_CUSTOM_STYLE_SOLID_COLOR,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_OBJECT (object_class, PROP_OUTLINE_PATTERN,
|
|
"outline-pattern", NULL, NULL,
|
|
PIKA_TYPE_PATTERN,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_RGB (object_class, PROP_OUTLINE_FOREGROUND,
|
|
"outline-foreground", NULL, NULL,
|
|
FALSE, &gray,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_OUTLINE_WIDTH,
|
|
"outline-width", NULL, NULL,
|
|
0.0, 8192.0, 4.0,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_DEFAULTS);
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_OUTLINE_CAP_STYLE,
|
|
"outline-cap-style", NULL, NULL,
|
|
PIKA_TYPE_CAP_STYLE, PIKA_CAP_BUTT,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_ENUM (object_class, PROP_OUTLINE_JOIN_STYLE,
|
|
"outline-join-style", NULL, NULL,
|
|
PIKA_TYPE_JOIN_STYLE, PIKA_JOIN_MITER,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_OUTLINE_MITER_LIMIT,
|
|
"outline-miter-limit",
|
|
NULL, NULL,
|
|
0.0, 100.0, 10.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_OUTLINE_ANTIALIAS,
|
|
"outline-antialias", NULL, NULL,
|
|
TRUE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
PIKA_CONFIG_PROP_DOUBLE (object_class, PROP_OUTLINE_DASH_OFFSET,
|
|
"outline-dash-offset", NULL, NULL,
|
|
0.0, 2000.0, 0.0,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
array_spec = g_param_spec_double ("outline-dash-length", NULL, NULL,
|
|
0.0, 2000.0, 1.0, PIKA_PARAM_READWRITE);
|
|
g_object_class_install_property (object_class, PROP_OUTLINE_DASH_INFO,
|
|
pika_param_spec_value_array ("outline-dash-info",
|
|
NULL, NULL,
|
|
array_spec,
|
|
PIKA_PARAM_STATIC_STRINGS |
|
|
PIKA_CONFIG_PARAM_FLAGS));
|
|
|
|
/* the old hinting options have been replaced by 'hint-style' */
|
|
PIKA_CONFIG_PROP_BOOLEAN (object_class, PROP_HINTING,
|
|
"hinting",
|
|
NULL, NULL,
|
|
TRUE,
|
|
PIKA_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_property (object_class, PROP_PIKA,
|
|
g_param_spec_object ("pika", NULL, NULL,
|
|
PIKA_TYPE_PIKA,
|
|
PIKA_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
}
|
|
|
|
static void
|
|
pika_text_init (PikaText *text)
|
|
{
|
|
}
|
|
|
|
static void
|
|
pika_text_config_iface_init (PikaConfigInterface *iface)
|
|
{
|
|
iface->serialize_property = pika_text_serialize_property;
|
|
iface->deserialize_property = pika_text_deserialize_property;
|
|
}
|
|
|
|
static void
|
|
pika_text_finalize (GObject *object)
|
|
{
|
|
PikaText *text = PIKA_TEXT (object);
|
|
|
|
g_clear_pointer (&text->text, g_free);
|
|
g_clear_pointer (&text->markup, g_free);
|
|
g_clear_pointer (&text->language, g_free);
|
|
g_clear_object (&text->font);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
pika_text_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PikaText *text = PIKA_TEXT (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_TEXT:
|
|
g_value_set_string (value, text->text);
|
|
break;
|
|
case PROP_MARKUP:
|
|
g_value_set_string (value, text->markup);
|
|
break;
|
|
case PROP_FONT:
|
|
g_value_set_object (value, text->font);
|
|
break;
|
|
case PROP_FONT_SIZE:
|
|
g_value_set_double (value, text->font_size);
|
|
break;
|
|
case PROP_UNIT:
|
|
g_value_set_int (value, text->unit);
|
|
break;
|
|
case PROP_ANTIALIAS:
|
|
g_value_set_boolean (value, text->antialias);
|
|
break;
|
|
case PROP_HINT_STYLE:
|
|
g_value_set_enum (value, text->hint_style);
|
|
break;
|
|
case PROP_KERNING:
|
|
g_value_set_boolean (value, text->kerning);
|
|
break;
|
|
case PROP_BASE_DIR:
|
|
g_value_set_enum (value, text->base_dir);
|
|
break;
|
|
case PROP_LANGUAGE:
|
|
g_value_set_string (value, text->language);
|
|
break;
|
|
case PROP_COLOR:
|
|
g_value_set_boxed (value, &text->color);
|
|
break;
|
|
case PROP_OUTLINE:
|
|
g_value_set_enum (value, text->outline);
|
|
break;
|
|
case PROP_JUSTIFICATION:
|
|
g_value_set_enum (value, text->justify);
|
|
break;
|
|
case PROP_INDENTATION:
|
|
g_value_set_double (value, text->indent);
|
|
break;
|
|
case PROP_LINE_SPACING:
|
|
g_value_set_double (value, text->line_spacing);
|
|
break;
|
|
case PROP_LETTER_SPACING:
|
|
g_value_set_double (value, text->letter_spacing);
|
|
break;
|
|
case PROP_BOX_MODE:
|
|
g_value_set_enum (value, text->box_mode);
|
|
break;
|
|
case PROP_BOX_WIDTH:
|
|
g_value_set_double (value, text->box_width);
|
|
break;
|
|
case PROP_BOX_HEIGHT:
|
|
g_value_set_double (value, text->box_height);
|
|
break;
|
|
case PROP_BOX_UNIT:
|
|
g_value_set_int (value, text->box_unit);
|
|
break;
|
|
case PROP_TRANSFORMATION:
|
|
g_value_set_boxed (value, &text->transformation);
|
|
break;
|
|
case PROP_OFFSET_X:
|
|
g_value_set_double (value, text->offset_x);
|
|
break;
|
|
case PROP_OFFSET_Y:
|
|
g_value_set_double (value, text->offset_y);
|
|
break;
|
|
case PROP_OUTLINE_STYLE:
|
|
g_value_set_enum (value, text->outline_style);
|
|
break;
|
|
case PROP_OUTLINE_FOREGROUND:
|
|
g_value_set_boxed (value, &text->outline_foreground);
|
|
break;
|
|
case PROP_OUTLINE_PATTERN:
|
|
g_value_set_object (value, text->outline_pattern);
|
|
break;
|
|
case PROP_OUTLINE_WIDTH:
|
|
g_value_set_double (value, text->outline_width);
|
|
break;
|
|
case PROP_OUTLINE_CAP_STYLE:
|
|
g_value_set_enum (value, text->outline_cap_style);
|
|
break;
|
|
case PROP_OUTLINE_JOIN_STYLE:
|
|
g_value_set_enum (value, text->outline_join_style);
|
|
break;
|
|
case PROP_OUTLINE_MITER_LIMIT:
|
|
g_value_set_double (value, text->outline_miter_limit);
|
|
break;
|
|
case PROP_OUTLINE_ANTIALIAS:
|
|
g_value_set_boolean (value, text->outline_antialias);
|
|
break;
|
|
case PROP_OUTLINE_DASH_OFFSET:
|
|
g_value_set_double (value, text->outline_dash_offset);
|
|
break;
|
|
case PROP_OUTLINE_DASH_INFO:
|
|
{
|
|
PikaValueArray *value_array;
|
|
|
|
value_array = pika_dash_pattern_to_value_array (text->outline_dash_info);
|
|
g_value_take_boxed (value, value_array);
|
|
}
|
|
break;
|
|
case PROP_HINTING:
|
|
g_value_set_boolean (value,
|
|
text->hint_style != PIKA_TEXT_HINT_STYLE_NONE);
|
|
break;
|
|
case PROP_PIKA:
|
|
g_value_set_object (value, text->pika);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_text_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
PikaText *text = PIKA_TEXT (object);
|
|
PikaRGB *color;
|
|
PikaMatrix2 *matrix;
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_TEXT:
|
|
g_free (text->text);
|
|
text->text = g_value_dup_string (value);
|
|
if (text->text && text->markup)
|
|
{
|
|
g_clear_pointer (&text->markup, g_free);
|
|
g_object_notify (object, "markup");
|
|
}
|
|
break;
|
|
case PROP_MARKUP:
|
|
g_free (text->markup);
|
|
text->markup = g_value_dup_string (value);
|
|
if (text->markup && text->text)
|
|
{
|
|
g_clear_pointer (&text->text, g_free);
|
|
g_object_notify (object, "text");
|
|
}
|
|
break;
|
|
case PROP_FONT:
|
|
{
|
|
PikaFont *font = g_value_get_object (value);
|
|
|
|
if (font != text->font)
|
|
g_set_object (&text->font, font);
|
|
}
|
|
break;
|
|
case PROP_FONT_SIZE:
|
|
text->font_size = g_value_get_double (value);
|
|
break;
|
|
case PROP_UNIT:
|
|
text->unit = g_value_get_int (value);
|
|
break;
|
|
case PROP_ANTIALIAS:
|
|
text->antialias = g_value_get_boolean (value);
|
|
break;
|
|
case PROP_HINT_STYLE:
|
|
text->hint_style = g_value_get_enum (value);
|
|
break;
|
|
case PROP_KERNING:
|
|
text->kerning = g_value_get_boolean (value);
|
|
break;
|
|
case PROP_LANGUAGE:
|
|
g_free (text->language);
|
|
text->language = g_value_dup_string (value);
|
|
break;
|
|
case PROP_BASE_DIR:
|
|
text->base_dir = g_value_get_enum (value);
|
|
break;
|
|
case PROP_COLOR:
|
|
color = g_value_get_boxed (value);
|
|
text->color = *color;
|
|
break;
|
|
case PROP_OUTLINE:
|
|
text->outline = g_value_get_enum (value);
|
|
break;
|
|
case PROP_JUSTIFICATION:
|
|
text->justify = g_value_get_enum (value);
|
|
break;
|
|
case PROP_INDENTATION:
|
|
text->indent = g_value_get_double (value);
|
|
break;
|
|
case PROP_LINE_SPACING:
|
|
text->line_spacing = g_value_get_double (value);
|
|
break;
|
|
case PROP_LETTER_SPACING:
|
|
text->letter_spacing = g_value_get_double (value);
|
|
break;
|
|
case PROP_BOX_MODE:
|
|
text->box_mode = g_value_get_enum (value);
|
|
break;
|
|
case PROP_BOX_WIDTH:
|
|
text->box_width = g_value_get_double (value);
|
|
break;
|
|
case PROP_BOX_HEIGHT:
|
|
text->box_height = g_value_get_double (value);
|
|
break;
|
|
case PROP_BOX_UNIT:
|
|
text->box_unit = g_value_get_int (value);
|
|
break;
|
|
case PROP_TRANSFORMATION:
|
|
matrix = g_value_get_boxed (value);
|
|
text->transformation = *matrix;
|
|
break;
|
|
case PROP_OFFSET_X:
|
|
text->offset_x = g_value_get_double (value);
|
|
break;
|
|
case PROP_OFFSET_Y:
|
|
text->offset_y = g_value_get_double (value);
|
|
break;
|
|
case PROP_OUTLINE_STYLE:
|
|
text->outline_style = g_value_get_enum (value);
|
|
break;
|
|
case PROP_OUTLINE_FOREGROUND:
|
|
color = g_value_get_boxed (value);
|
|
text->outline_foreground = *color;
|
|
break;
|
|
case PROP_OUTLINE_PATTERN:
|
|
{
|
|
PikaPattern *pattern = g_value_get_object (value);
|
|
|
|
if (text->outline_pattern != pattern)
|
|
{
|
|
if (text->outline_pattern)
|
|
g_object_unref (text->outline_pattern);
|
|
|
|
text->outline_pattern = pattern ? g_object_ref (pattern) : pattern;
|
|
}
|
|
break;
|
|
}
|
|
case PROP_OUTLINE_WIDTH:
|
|
text->outline_width = g_value_get_double (value);
|
|
break;
|
|
case PROP_OUTLINE_CAP_STYLE:
|
|
text->outline_cap_style = g_value_get_enum (value);
|
|
break;
|
|
case PROP_OUTLINE_JOIN_STYLE:
|
|
text->outline_join_style = g_value_get_enum (value);
|
|
break;
|
|
case PROP_OUTLINE_MITER_LIMIT:
|
|
text->outline_miter_limit = g_value_get_double (value);
|
|
break;
|
|
case PROP_OUTLINE_ANTIALIAS:
|
|
text->outline_antialias = g_value_get_boolean (value);
|
|
break;
|
|
case PROP_OUTLINE_DASH_OFFSET:
|
|
text->outline_dash_offset = g_value_get_double (value);
|
|
break;
|
|
case PROP_OUTLINE_DASH_INFO:
|
|
{
|
|
PikaValueArray *value_array = g_value_get_boxed (value);
|
|
text->outline_dash_info = pika_dash_pattern_from_value_array (value_array);
|
|
}
|
|
break;
|
|
case PROP_BORDER:
|
|
text->border = g_value_get_int (value);
|
|
break;
|
|
case PROP_HINTING:
|
|
/* interpret "hinting" only if "hint-style" has its default
|
|
* value, so we don't overwrite a serialized new hint-style with
|
|
* a compat "hinting" that is only there for old PIKA versions
|
|
*/
|
|
if (text->hint_style == PIKA_TEXT_HINT_STYLE_MEDIUM)
|
|
text->hint_style = (g_value_get_boolean (value) ?
|
|
PIKA_TEXT_HINT_STYLE_MEDIUM :
|
|
PIKA_TEXT_HINT_STYLE_NONE);
|
|
break;
|
|
case PROP_PIKA:
|
|
text->pika = g_value_get_object (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_text_dispatch_properties_changed (GObject *object,
|
|
guint n_pspecs,
|
|
GParamSpec **pspecs)
|
|
{
|
|
G_OBJECT_CLASS (parent_class)->dispatch_properties_changed (object,
|
|
n_pspecs, pspecs);
|
|
|
|
g_signal_emit (object, text_signals[CHANGED], 0);
|
|
}
|
|
|
|
static gint64
|
|
pika_text_get_memsize (PikaObject *object,
|
|
gint64 *gui_size)
|
|
{
|
|
PikaText *text = PIKA_TEXT (object);
|
|
gint64 memsize = 0;
|
|
|
|
memsize += pika_string_get_memsize (text->text);
|
|
memsize += pika_string_get_memsize (text->markup);
|
|
memsize += pika_string_get_memsize (text->language);
|
|
|
|
return memsize + PIKA_OBJECT_CLASS (parent_class)->get_memsize (object,
|
|
gui_size);
|
|
}
|
|
|
|
void
|
|
pika_text_get_transformation (PikaText *text,
|
|
PikaMatrix3 *matrix)
|
|
{
|
|
g_return_if_fail (PIKA_IS_TEXT (text));
|
|
g_return_if_fail (matrix != NULL);
|
|
|
|
matrix->coeff[0][0] = text->transformation.coeff[0][0];
|
|
matrix->coeff[0][1] = text->transformation.coeff[0][1];
|
|
matrix->coeff[0][2] = text->offset_x;
|
|
|
|
matrix->coeff[1][0] = text->transformation.coeff[1][0];
|
|
matrix->coeff[1][1] = text->transformation.coeff[1][1];
|
|
matrix->coeff[1][2] = text->offset_y;
|
|
|
|
matrix->coeff[2][0] = 0.0;
|
|
matrix->coeff[2][1] = 0.0;
|
|
matrix->coeff[2][2] = 1.0;
|
|
}
|
|
|
|
static gboolean
|
|
pika_text_serialize_property (PikaConfig *config,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec,
|
|
PikaConfigWriter *writer)
|
|
{
|
|
if (property_id == PROP_OUTLINE_PATTERN)
|
|
{
|
|
PikaObject *serialize_obj = g_value_get_object (value);
|
|
|
|
pika_config_writer_open (writer, pspec->name);
|
|
|
|
if (serialize_obj)
|
|
pika_config_writer_string (writer, pika_object_get_name (serialize_obj));
|
|
else
|
|
pika_config_writer_print (writer, "NULL", 4);
|
|
|
|
pika_config_writer_close (writer);
|
|
|
|
return TRUE;
|
|
}
|
|
else if (property_id == PROP_MARKUP)
|
|
{
|
|
gchar *markup = (gchar*)g_value_get_string (value);
|
|
GRegex *regex;
|
|
PikaText *text;
|
|
PikaContainer *container;
|
|
PangoAttrList *attr_list;
|
|
PikaFont *font;
|
|
guint length;
|
|
GSList *list = NULL;
|
|
GSList *fonts = NULL;
|
|
|
|
g_return_val_if_fail (PIKA_IS_TEXT (config), FALSE);
|
|
|
|
if (markup == NULL)
|
|
return FALSE;
|
|
|
|
text = PIKA_TEXT (config);
|
|
container = pika_data_factory_get_container (text->pika->font_factory);
|
|
|
|
/*lookupname format is "pikafont%d" we keep only the "font%d" part
|
|
* this is to avoid problems when deserializing
|
|
* e.g. if there are 2 fonts with lookupname pikafont17 and pikafont23
|
|
* we might replace the first with a font whose lookupname is pikafont23,
|
|
* and we might replace the original pikafont23 with say pikafont29
|
|
* this means that all occurences of pikafont23 turned into pikafont29
|
|
*/
|
|
regex = g_regex_new ("\"pikafont(\\d+)\"", 0, 0, NULL);
|
|
markup = g_regex_replace (regex, markup, -1, 0, "\"font\\1\"", 0, NULL);
|
|
|
|
pika_config_writer_open (writer, "markup");
|
|
pika_config_writer_string (writer, markup);
|
|
|
|
pango_parse_markup (markup, -1, 0, &attr_list, NULL, NULL, NULL);
|
|
|
|
list = pango_attr_list_get_attributes (attr_list);
|
|
length = g_slist_length (list);
|
|
|
|
for (guint i = 0; i < length; ++i)
|
|
{
|
|
PangoAttrFontDesc *attr_font_desc = pango_attribute_as_font_desc ((PangoAttribute*)g_slist_nth_data (list, i));
|
|
|
|
if (attr_font_desc != NULL)
|
|
{
|
|
gchar *altered_font_name = pango_font_description_to_string (attr_font_desc->desc);
|
|
gchar *font_name = g_strdup_printf ("pika%s", altered_font_name);
|
|
|
|
if (g_slist_find_custom (fonts, (gconstpointer) font_name, (GCompareFunc) g_strcmp0) == NULL)
|
|
{
|
|
fonts = g_slist_prepend (fonts, (gpointer) font_name);
|
|
|
|
font = PIKA_FONT (pika_container_search (container,
|
|
(PikaContainerSearchFunc) pika_font_match_by_lookup_name,
|
|
(gpointer) font_name));
|
|
|
|
pika_config_writer_open (writer, "markupfont");
|
|
/*lookupname format is "font%d" we keep only the "font%d" (see the above comment)*/
|
|
pika_config_writer_string (writer, font_name+4);
|
|
|
|
pika_config_writer_open (writer, "font");
|
|
PIKA_CONFIG_GET_IFACE (PIKA_CONFIG (font))->serialize (PIKA_CONFIG (font),
|
|
writer,
|
|
NULL);
|
|
pika_config_writer_close (writer);
|
|
pika_config_writer_close (writer);
|
|
}
|
|
|
|
else
|
|
{
|
|
g_free (font_name);
|
|
}
|
|
g_free (altered_font_name);
|
|
}
|
|
}
|
|
|
|
pika_config_writer_close (writer);
|
|
|
|
g_slist_free_full (fonts, (GDestroyNotify) g_free);
|
|
g_slist_free_full (list, (GDestroyNotify) pango_attribute_destroy);
|
|
pango_attr_list_unref (attr_list);
|
|
g_free (markup);
|
|
g_regex_unref (regex);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
pika_text_deserialize_property (PikaConfig *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec,
|
|
GScanner *scanner,
|
|
GTokenType *expected)
|
|
{
|
|
if (property_id == PROP_OUTLINE_PATTERN)
|
|
{
|
|
gchar *object_name;
|
|
|
|
if (pika_scanner_parse_identifier (scanner, "NULL"))
|
|
{
|
|
g_value_set_object (value, NULL);
|
|
}
|
|
else if (pika_scanner_parse_string (scanner, &object_name))
|
|
{
|
|
PikaText *text = PIKA_TEXT (object);
|
|
PikaContainer *container;
|
|
PikaObject *deserialize_obj;
|
|
|
|
if (! object_name)
|
|
object_name = g_strdup ("");
|
|
|
|
container = pika_data_factory_get_container (text->pika->pattern_factory);
|
|
|
|
deserialize_obj = pika_container_get_child_by_name (container,
|
|
object_name);
|
|
|
|
g_value_set_object (value, deserialize_obj);
|
|
|
|
g_free (object_name);
|
|
}
|
|
else
|
|
{
|
|
*expected = G_TOKEN_STRING;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
else if (property_id == PROP_OUTLINE_DASH_INFO)
|
|
{
|
|
if (pika_scanner_parse_identifier (scanner, "NULL"))
|
|
{
|
|
g_value_take_boxed (value, NULL);
|
|
return TRUE;
|
|
}
|
|
}
|
|
else if (property_id == PROP_MARKUP)
|
|
{
|
|
gchar *markup = NULL;
|
|
GString *markup_str;
|
|
PikaFont *dummy_object = g_object_new (PIKA_TYPE_FONT, NULL);
|
|
|
|
pika_scanner_parse_string (scanner, &markup);
|
|
|
|
markup_str = g_string_new (markup);
|
|
|
|
/* This is for backward compatibility with older xcf files.*/
|
|
if (g_scanner_peek_next_token (scanner) == G_TOKEN_STRING)
|
|
{
|
|
while (g_scanner_peek_next_token (scanner) == G_TOKEN_STRING)
|
|
{
|
|
gchar *markup_fontname = NULL;
|
|
gchar *replaced_markup;
|
|
gchar *new_markup;
|
|
PikaFont *font;
|
|
|
|
pika_scanner_parse_string (scanner, &markup_fontname);
|
|
|
|
font = PIKA_FONT (PIKA_CONFIG_GET_IFACE (dummy_object)->deserialize_create (PIKA_TYPE_FONT,
|
|
scanner,
|
|
-1,
|
|
NULL));
|
|
replaced_markup = g_markup_printf_escaped (" font=\"%s\"", markup_fontname);
|
|
new_markup = g_strdup_printf (" pikafont=\"%s\"", pika_font_get_lookup_name (font));
|
|
g_string_replace (markup_str, replaced_markup, new_markup, 0);
|
|
|
|
g_free (markup_fontname);
|
|
g_free (replaced_markup);
|
|
g_free (new_markup);
|
|
g_object_unref (font);
|
|
}
|
|
/* We avoid the edge case when actual fonts are called "pikafont%d"
|
|
* and their replacement name chains to each other by marking already
|
|
* processed font as "pikafont=". Then we clean up this fake attribute
|
|
* name in the end.
|
|
* This is not a problem with the new format as font are stored as
|
|
* "font%d" (see comment in pika_text_serialize_property()).
|
|
*/
|
|
g_string_replace (markup_str, " pikafont=\"", " font=\"", 0);
|
|
}
|
|
else
|
|
{
|
|
while (g_scanner_peek_next_token (scanner) == G_TOKEN_LEFT_PAREN)
|
|
{
|
|
gchar *lookupname = NULL;
|
|
gchar *replaced_markup;
|
|
gchar *new_markup;
|
|
PikaFont *font;
|
|
|
|
g_scanner_get_next_token (scanner); /* ( */
|
|
g_scanner_get_next_token (scanner); /* "lookupname" */
|
|
pika_scanner_parse_string (scanner, &lookupname);
|
|
|
|
g_scanner_get_next_token (scanner); /* ) */
|
|
g_scanner_get_next_token (scanner); /* font */
|
|
|
|
font = PIKA_FONT (PIKA_CONFIG_GET_IFACE (dummy_object)->deserialize_create (PIKA_TYPE_FONT,
|
|
scanner,
|
|
-1,
|
|
NULL));
|
|
g_scanner_get_next_token (scanner); /* ) */
|
|
g_scanner_get_next_token (scanner); /* ) */
|
|
|
|
replaced_markup = g_markup_printf_escaped (" font=\"%s\"", lookupname);
|
|
new_markup = g_strdup_printf (" font=\"%s\"", pika_font_get_lookup_name (font));
|
|
g_string_replace (markup_str, replaced_markup, new_markup, 0);
|
|
|
|
g_free (lookupname);
|
|
g_free (replaced_markup);
|
|
g_free (new_markup);
|
|
g_object_unref (font);
|
|
}
|
|
}
|
|
|
|
g_value_set_string (value, markup_str->str);
|
|
|
|
g_object_unref (dummy_object);
|
|
g_string_free (markup_str, TRUE);
|
|
g_free (markup);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|