Files
gtk/ottie/ottiegroupshape.c
Matthias Clasen b558107dbe ottie: Add printing
Add and implement ottie_object_print and ottie_creation_to_bytes.
2020-12-28 08:45:49 -05:00

282 lines
7.7 KiB
C

/*
* Copyright © 2020 Benjamin Otte
*
* 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, see <http://www.gnu.org/licenses/>.
*
* Authors: Benjamin Otte <otte@gnome.org>
*/
#include "config.h"
#include "ottiegroupshapeprivate.h"
#include "ottieellipseshapeprivate.h"
#include "ottiefillshapeprivate.h"
#include "ottieparserprivate.h"
#include "ottiepathshapeprivate.h"
#include "ottierectshapeprivate.h"
#include "ottieshapeprivate.h"
#include "ottiestrokeshapeprivate.h"
#include "ottietransformprivate.h"
#include "ottietrimshapeprivate.h"
#include <glib/gi18n-lib.h>
#include <gsk/gsk.h>
#define GDK_ARRAY_ELEMENT_TYPE OttieShape *
#define GDK_ARRAY_FREE_FUNC g_object_unref
#define GDK_ARRAY_TYPE_NAME OttieShapeList
#define GDK_ARRAY_NAME ottie_shape_list
#define GDK_ARRAY_PREALLOC 4
#include "gdk/gdkarrayimpl.c"
struct _OttieGroupShape
{
OttieShape parent;
OttieShapeList shapes;
GskBlendMode blend_mode;
};
struct _OttieGroupShapeClass
{
OttieShapeClass parent_class;
};
static GType
ottie_group_shape_get_item_type (GListModel *list)
{
return OTTIE_TYPE_SHAPE;
}
static guint
ottie_group_shape_get_n_items (GListModel *list)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (list);
return ottie_shape_list_get_size (&self->shapes);
}
static gpointer
ottie_group_shape_get_item (GListModel *list,
guint position)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (list);
if (position >= ottie_shape_list_get_size (&self->shapes))
return NULL;
return g_object_ref (ottie_shape_list_get (&self->shapes, position));
}
static void
ottie_group_shape_list_model_init (GListModelInterface *iface)
{
iface->get_item_type = ottie_group_shape_get_item_type;
iface->get_n_items = ottie_group_shape_get_n_items;
iface->get_item = ottie_group_shape_get_item;
}
G_DEFINE_TYPE_WITH_CODE (OttieGroupShape, ottie_group_shape, OTTIE_TYPE_SHAPE,
G_IMPLEMENT_INTERFACE (G_TYPE_LIST_MODEL, ottie_group_shape_list_model_init))
static void
ottie_group_shape_render (OttieShape *shape,
OttieRender *render,
double timestamp)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (shape);
OttieRender child_render;
ottie_render_init_child (&child_render, render);
for (gsize i = 0; i < ottie_shape_list_get_size (&self->shapes); i++)
{
ottie_shape_render (ottie_shape_list_get (&self->shapes, i),
&child_render,
timestamp);
}
ottie_render_merge (render, &child_render);
ottie_render_clear (&child_render);
}
static void
ottie_group_shape_print (OttieObject *obj,
OttiePrinter *printer)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (obj);
OTTIE_OBJECT_CLASS (ottie_group_shape_parent_class)->print (obj, printer);
ottie_printer_add_string (printer, "ty", "gr");
ottie_printer_add_int (printer, "bm", self->blend_mode);
ottie_group_shape_print_shapes (OTTIE_SHAPE (self), "it", printer);
}
static void
ottie_group_shape_dispose (GObject *object)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (object);
ottie_shape_list_clear (&self->shapes);
G_OBJECT_CLASS (ottie_group_shape_parent_class)->dispose (object);
}
static void
ottie_group_shape_class_init (OttieGroupShapeClass *klass)
{
OttieObjectClass *oobject_class = OTTIE_OBJECT_CLASS (klass);
OttieShapeClass *shape_class = OTTIE_SHAPE_CLASS (klass);
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
oobject_class->print = ottie_group_shape_print;
shape_class->render = ottie_group_shape_render;
gobject_class->dispose = ottie_group_shape_dispose;
}
static void
ottie_group_shape_init (OttieGroupShape *self)
{
}
gboolean
ottie_group_shape_parse_shapes (JsonReader *reader,
gsize offset,
gpointer data)
{
OttieGroupShape *self = data;
if (!json_reader_is_array (reader))
{
ottie_parser_error_syntax (reader, "Shapes are not an array.");
return FALSE;
}
for (int i = 0; ; i++)
{
OttieShape *shape;
const char *type;
if (!json_reader_read_element (reader, i))
break;
if (!json_reader_is_object (reader))
{
ottie_parser_error_syntax (reader, "Shape %d is not an object", i);
continue;
}
if (!json_reader_read_member (reader, "ty"))
{
ottie_parser_error_syntax (reader, "Shape %d has no type", i);
json_reader_end_member (reader);
json_reader_end_element (reader);
continue;
}
type = json_reader_get_string_value (reader);
if (type == NULL || json_reader_get_error (reader))
{
ottie_parser_emit_error (reader, json_reader_get_error (reader));
json_reader_end_member (reader);
json_reader_end_element (reader);
continue;
}
json_reader_end_member (reader);
if (g_str_equal (type, "el"))
shape = ottie_ellipse_shape_parse (reader);
else if (g_str_equal (type, "fl"))
shape = ottie_fill_shape_parse (reader);
else if (g_str_equal (type, "gr"))
shape = ottie_group_shape_parse (reader);
else if (g_str_equal (type, "rc"))
shape = ottie_rect_shape_parse (reader);
else if (g_str_equal (type, "sh"))
shape = ottie_path_shape_parse (reader);
else if (g_str_equal (type, "st"))
shape = ottie_stroke_shape_parse (reader);
else if (g_str_equal (type, "tm"))
shape = ottie_trim_shape_parse (reader);
else if (g_str_equal (type, "tr"))
shape = ottie_transform_parse (reader);
else
{
ottie_parser_error_value (reader, "Shape %d has unknown type \"%s\"", i, type);
shape = NULL;
}
if (shape)
ottie_shape_list_append (&self->shapes, shape);
json_reader_end_element (reader);
}
json_reader_end_element (reader);
return TRUE;
}
OttieShape *
ottie_group_shape_new (void)
{
return g_object_new (OTTIE_TYPE_GROUP_SHAPE, NULL);
}
OttieShape *
ottie_group_shape_parse (JsonReader *reader)
{
OttieParserOption options[] = {
OTTIE_PARSE_OPTIONS_SHAPE,
{ "bm", ottie_parser_option_blend_mode, G_STRUCT_OFFSET (OttieGroupShape, blend_mode) },
{ "np", ottie_parser_option_skip_expression, 0 },
{ "cix", ottie_parser_option_skip_index, 0 },
{ "it", ottie_group_shape_parse_shapes, 0 },
};
OttieShape *self;
self = ottie_group_shape_new ();
if (!ottie_parser_parse_object (reader, "group shape", options, G_N_ELEMENTS (options), self))
{
g_object_unref (self);
return NULL;
}
return self;
}
void
ottie_group_shape_print_shapes (OttieShape *shape,
const char *name,
OttiePrinter *printer)
{
OttieGroupShape *self = OTTIE_GROUP_SHAPE (shape);
ottie_printer_start_array (printer, name);
for (gsize i = 0; i < ottie_shape_list_get_size (&self->shapes); i++)
{
OttieObject *obj = OTTIE_OBJECT (ottie_shape_list_get (&self->shapes, i));
ottie_object_print (obj, NULL, printer);
}
ottie_printer_end_array (printer);
}