2010-04-08 09:55:15 +00:00
|
|
|
|
/*
|
|
|
|
|
* Clutter.
|
|
|
|
|
*
|
|
|
|
|
* An OpenGL based 'interactive canvas' library.
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2010 Intel Corporation.
|
|
|
|
|
*
|
|
|
|
|
* 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 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/>.
|
|
|
|
|
*
|
|
|
|
|
* Author:
|
|
|
|
|
* Emmanuele Bassi <ebassi@linux.intel.com>
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
2022-05-25 16:41:48 +00:00
|
|
|
|
* ClutterEffect:
|
|
|
|
|
*
|
|
|
|
|
* Base class for actor effects
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
|
|
|
|
* The #ClutterEffect class provides a default type and API for creating
|
|
|
|
|
* effects for generic actors.
|
|
|
|
|
*
|
|
|
|
|
* Effects are a #ClutterActorMeta sub-class that modify the way an actor
|
|
|
|
|
* is painted in a way that is not part of the actor's implementation.
|
|
|
|
|
*
|
|
|
|
|
* Effects should be the preferred way to affect the paint sequence of an
|
|
|
|
|
* actor without sub-classing the actor itself and overriding the
|
2024-01-04 13:40:48 +00:00
|
|
|
|
* [vfunc@Clutter.Actor.paint] virtual function.
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* ## Implementing a ClutterEffect
|
|
|
|
|
*
|
|
|
|
|
* Creating a sub-class of #ClutterEffect requires overriding the
|
2024-01-04 13:40:48 +00:00
|
|
|
|
* [vfunc@Clutter.Effect.paint] method. The implementation of the function should look
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* something like this:
|
|
|
|
|
*
|
2022-05-25 16:41:48 +00:00
|
|
|
|
* ```c
|
2011-06-13 14:58:46 +00:00
|
|
|
|
* void effect_paint (ClutterEffect *effect, ClutterEffectPaintFlags flags)
|
2011-03-02 16:55:10 +00:00
|
|
|
|
* {
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Set up initialisation of the paint such as binding a
|
|
|
|
|
* // CoglOffscreen or other operations
|
2011-03-02 16:55:10 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Chain to the next item in the paint sequence. This will either call
|
|
|
|
|
* // ‘paint’ on the next effect or just paint the actor if this is
|
|
|
|
|
* // the last effect.
|
2011-03-02 16:55:10 +00:00
|
|
|
|
* ClutterActor *actor =
|
|
|
|
|
* clutter_actor_meta_get_actor (CLUTTER_ACTOR_META (effect));
|
2014-03-17 23:07:58 +00:00
|
|
|
|
*
|
2011-03-02 16:55:10 +00:00
|
|
|
|
* clutter_actor_continue_paint (actor);
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // perform any cleanup of state, such as popping the CoglOffscreen
|
2011-03-02 16:55:10 +00:00
|
|
|
|
* }
|
2022-05-25 16:41:48 +00:00
|
|
|
|
* ```
|
2014-03-17 23:07:58 +00:00
|
|
|
|
*
|
|
|
|
|
* The effect can optionally avoid calling clutter_actor_continue_paint() to skip any
|
|
|
|
|
* further stages of the paint sequence. This is useful for example if the effect
|
|
|
|
|
* contains a cached image of the actor. In that case it can optimise painting by
|
|
|
|
|
* avoiding the actor paint and instead painting the cached image.
|
|
|
|
|
*
|
|
|
|
|
* The %CLUTTER_EFFECT_PAINT_ACTOR_DIRTY flag is useful in this case. Clutter will set
|
|
|
|
|
* this flag when a redraw has been queued on the actor since it was last painted. The
|
|
|
|
|
* effect can use this information to decide if the cached image is still valid.
|
|
|
|
|
*
|
|
|
|
|
* ## A simple ClutterEffect implementation
|
|
|
|
|
*
|
|
|
|
|
* The example below creates two rectangles: one will be painted "behind" the actor,
|
|
|
|
|
* while another will be painted "on top" of the actor.
|
|
|
|
|
*
|
|
|
|
|
* The #ClutterActorMetaClass.set_actor() implementation will create the two materials
|
|
|
|
|
* used for the two different rectangles; the #ClutterEffectClass.paint() implementation
|
|
|
|
|
* will paint the first material using cogl_rectangle(), before continuing and then it
|
|
|
|
|
* will paint paint the second material after.
|
|
|
|
|
*
|
2022-05-25 16:41:48 +00:00
|
|
|
|
* ```c
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* typedef struct {
|
|
|
|
|
* ClutterEffect parent_instance;
|
|
|
|
|
*
|
2023-08-18 17:39:49 +00:00
|
|
|
|
* CoglPipeline *rect_1;
|
|
|
|
|
* CoglPipeline *rect_2;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* } MyEffect;
|
|
|
|
|
*
|
|
|
|
|
* typedef struct _ClutterEffectClass MyEffectClass;
|
|
|
|
|
*
|
|
|
|
|
* G_DEFINE_TYPE (MyEffect, my_effect, CLUTTER_TYPE_EFFECT);
|
|
|
|
|
*
|
|
|
|
|
* static void
|
|
|
|
|
* my_effect_set_actor (ClutterActorMeta *meta,
|
|
|
|
|
* ClutterActor *actor)
|
|
|
|
|
* {
|
|
|
|
|
* MyEffect *self = MY_EFFECT (meta);
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Clear the previous state //
|
|
|
|
|
* if (self->rect_1)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* {
|
2023-09-19 09:16:43 +00:00
|
|
|
|
* g_object_unref (self->rect_1);
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* self->rect_1 = NULL;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* }
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* if (self->rect_2)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* {
|
2023-09-19 09:16:43 +00:00
|
|
|
|
* g_object_unref (self->rect_2);
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* self->rect_2 = NULL;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* }
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Maintain a pointer to the actor
|
|
|
|
|
* self->actor = actor;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // If we've been detached by the actor then we should just bail out here
|
|
|
|
|
* if (self->actor == NULL)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* return;
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Create a red material
|
2023-08-18 17:39:49 +00:00
|
|
|
|
* self->rect_1 = cogl_pipeline_new ();
|
|
|
|
|
* cogl_pipeline_set_color4f (self->rect_1, 1.0, 0.0, 0.0, 1.0);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Create a green material
|
2023-08-18 17:39:49 +00:00
|
|
|
|
* self->rect_2 = cogl_pipeline_new ();
|
|
|
|
|
* cogl_pipeline_set_color4f (self->rect_2, 0.0, 1.0, 0.0, 1.0);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* static gboolean
|
2011-06-06 06:44:11 +00:00
|
|
|
|
* my_effect_paint (ClutterEffect *effect)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* {
|
|
|
|
|
* MyEffect *self = MY_EFFECT (effect);
|
|
|
|
|
* gfloat width, height;
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* clutter_actor_get_size (self->actor, &width, &height);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Paint the first rectangle in the upper left quadrant
|
|
|
|
|
* cogl_set_source (self->rect_1);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* cogl_rectangle (0, 0, width / 2, height / 2);
|
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Continue to the rest of the paint sequence
|
|
|
|
|
* clutter_actor_continue_paint (self->actor);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* // Paint the second rectangle in the lower right quadrant
|
|
|
|
|
* cogl_set_source (self->rect_2);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* cogl_rectangle (width / 2, height / 2, width, height);
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* static void
|
|
|
|
|
* my_effect_class_init (MyEffectClass *klass)
|
|
|
|
|
* {
|
2023-08-09 23:44:52 +00:00
|
|
|
|
* ClutterActorMetaClass *meta_class = CLUTTER_ACTOR_META_CLASS (klass);
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* meta_class->set_actor = my_effect_set_actor;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*
|
2014-03-17 23:07:58 +00:00
|
|
|
|
* klass->paint = my_effect_paint;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
* }
|
2022-05-25 16:41:48 +00:00
|
|
|
|
* ```
|
2010-04-08 09:55:15 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2023-11-07 10:56:00 +00:00
|
|
|
|
#include "config.h"
|
2010-04-08 09:55:15 +00:00
|
|
|
|
|
2023-08-07 13:38:12 +00:00
|
|
|
|
#include "clutter/clutter-effect.h"
|
2010-04-08 09:55:15 +00:00
|
|
|
|
|
2023-08-07 13:38:12 +00:00
|
|
|
|
#include "clutter/clutter-actor-meta-private.h"
|
|
|
|
|
#include "clutter/clutter-debug.h"
|
|
|
|
|
#include "clutter/clutter-effect-private.h"
|
|
|
|
|
#include "clutter/clutter-enum-types.h"
|
|
|
|
|
#include "clutter/clutter-marshal.h"
|
|
|
|
|
#include "clutter/clutter-paint-node-private.h"
|
|
|
|
|
#include "clutter/clutter-paint-nodes.h"
|
|
|
|
|
#include "clutter/clutter-private.h"
|
|
|
|
|
#include "clutter/clutter-actor-private.h"
|
2010-04-08 09:55:15 +00:00
|
|
|
|
|
|
|
|
|
G_DEFINE_ABSTRACT_TYPE (ClutterEffect,
|
|
|
|
|
clutter_effect,
|
|
|
|
|
CLUTTER_TYPE_ACTOR_META);
|
|
|
|
|
|
|
|
|
|
static gboolean
|
2019-11-13 21:21:58 +00:00
|
|
|
|
clutter_effect_real_pre_paint (ClutterEffect *effect,
|
2020-07-05 21:45:03 +00:00
|
|
|
|
ClutterPaintNode *node,
|
2019-11-13 21:21:58 +00:00
|
|
|
|
ClutterPaintContext *paint_context)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
{
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2019-11-13 21:21:58 +00:00
|
|
|
|
clutter_effect_real_post_paint (ClutterEffect *effect,
|
2020-07-05 21:45:03 +00:00
|
|
|
|
ClutterPaintNode *node,
|
2019-11-13 21:21:58 +00:00
|
|
|
|
ClutterPaintContext *paint_context)
|
2010-04-08 09:55:15 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-07 17:04:19 +00:00
|
|
|
|
static gboolean
|
2018-05-23 06:46:40 +00:00
|
|
|
|
clutter_effect_real_modify_paint_volume (ClutterEffect *effect,
|
|
|
|
|
ClutterPaintVolume *volume)
|
2010-08-16 16:02:15 +00:00
|
|
|
|
{
|
2010-09-07 17:04:19 +00:00
|
|
|
|
return TRUE;
|
2010-08-16 16:02:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
clutter/effect: Add paint_node vfunc
Introduce a new paint_node vfunc that, if implemented, allows
the effect to add nodes to a transient paint node that is
immediately painted. This is a transitional step until we
have fully delegated paint node rendering.
The most basic implementation of a ClutterEffect.paint_node
vfunc, and also the default implementation, is with an actor
node, as follows:
```
static void
foo_bar_paint_node (ClutterEffect *effect,
ClutterPaintNode *node,
ClutterPaintContext *paint_context,
ClutterEffectPaintFlags flags)
{
g_autoptr (ClutterPaintNode) actor_node = NULL;
actor_node = clutter_actor_node_new (effect->actor);
clutter_paint_node_add_child (node, actor_node);
}
```
This example gives the exact same behavior of simply calling
clutter_actor_continue_paint(). In the future, the paint node
itself will be a parameter of clutter_actor_continue_paint()
and we'll be able to simplify it event more.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1340>
2020-06-27 15:12:03 +00:00
|
|
|
|
static void
|
2020-12-29 17:43:22 +00:00
|
|
|
|
add_actor_node (ClutterEffect *effect,
|
|
|
|
|
ClutterPaintNode *node)
|
clutter/effect: Add paint_node vfunc
Introduce a new paint_node vfunc that, if implemented, allows
the effect to add nodes to a transient paint node that is
immediately painted. This is a transitional step until we
have fully delegated paint node rendering.
The most basic implementation of a ClutterEffect.paint_node
vfunc, and also the default implementation, is with an actor
node, as follows:
```
static void
foo_bar_paint_node (ClutterEffect *effect,
ClutterPaintNode *node,
ClutterPaintContext *paint_context,
ClutterEffectPaintFlags flags)
{
g_autoptr (ClutterPaintNode) actor_node = NULL;
actor_node = clutter_actor_node_new (effect->actor);
clutter_paint_node_add_child (node, actor_node);
}
```
This example gives the exact same behavior of simply calling
clutter_actor_continue_paint(). In the future, the paint node
itself will be a parameter of clutter_actor_continue_paint()
and we'll be able to simplify it event more.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1340>
2020-06-27 15:12:03 +00:00
|
|
|
|
{
|
|
|
|
|
ClutterPaintNode *actor_node;
|
|
|
|
|
ClutterActor *actor;
|
|
|
|
|
|
|
|
|
|
actor = clutter_actor_meta_get_actor (CLUTTER_ACTOR_META (effect));
|
|
|
|
|
|
2020-07-05 19:27:08 +00:00
|
|
|
|
actor_node = clutter_actor_node_new (actor, -1);
|
clutter/effect: Add paint_node vfunc
Introduce a new paint_node vfunc that, if implemented, allows
the effect to add nodes to a transient paint node that is
immediately painted. This is a transitional step until we
have fully delegated paint node rendering.
The most basic implementation of a ClutterEffect.paint_node
vfunc, and also the default implementation, is with an actor
node, as follows:
```
static void
foo_bar_paint_node (ClutterEffect *effect,
ClutterPaintNode *node,
ClutterPaintContext *paint_context,
ClutterEffectPaintFlags flags)
{
g_autoptr (ClutterPaintNode) actor_node = NULL;
actor_node = clutter_actor_node_new (effect->actor);
clutter_paint_node_add_child (node, actor_node);
}
```
This example gives the exact same behavior of simply calling
clutter_actor_continue_paint(). In the future, the paint node
itself will be a parameter of clutter_actor_continue_paint()
and we'll be able to simplify it event more.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1340>
2020-06-27 15:12:03 +00:00
|
|
|
|
clutter_paint_node_add_child (node, actor_node);
|
|
|
|
|
clutter_paint_node_unref (actor_node);
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-29 17:43:22 +00:00
|
|
|
|
static void
|
|
|
|
|
clutter_effect_real_paint_node (ClutterEffect *effect,
|
|
|
|
|
ClutterPaintNode *node,
|
|
|
|
|
ClutterPaintContext *paint_context,
|
|
|
|
|
ClutterEffectPaintFlags flags)
|
|
|
|
|
{
|
|
|
|
|
add_actor_node (effect, node);
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-02 16:55:10 +00:00
|
|
|
|
static void
|
2011-06-13 14:58:46 +00:00
|
|
|
|
clutter_effect_real_paint (ClutterEffect *effect,
|
2020-07-05 21:45:03 +00:00
|
|
|
|
ClutterPaintNode *node,
|
2019-11-13 21:21:58 +00:00
|
|
|
|
ClutterPaintContext *paint_context,
|
2011-06-13 14:58:46 +00:00
|
|
|
|
ClutterEffectPaintFlags flags)
|
2011-03-02 16:55:10 +00:00
|
|
|
|
{
|
2020-06-27 02:11:36 +00:00
|
|
|
|
ClutterEffectClass *effect_class = CLUTTER_EFFECT_GET_CLASS (effect);
|
2011-03-02 16:55:10 +00:00
|
|
|
|
gboolean pre_paint_succeeded;
|
|
|
|
|
|
|
|
|
|
/* The default implementation provides a compatibility wrapper for
|
2011-06-06 06:44:11 +00:00
|
|
|
|
effects that haven't migrated to use the 'paint' virtual yet. This
|
2011-03-02 16:55:10 +00:00
|
|
|
|
just calls the old pre and post virtuals before chaining on */
|
|
|
|
|
|
2020-07-05 21:45:03 +00:00
|
|
|
|
pre_paint_succeeded = effect_class->pre_paint (effect, node,paint_context);
|
clutter/effect: Add paint_node vfunc
Introduce a new paint_node vfunc that, if implemented, allows
the effect to add nodes to a transient paint node that is
immediately painted. This is a transitional step until we
have fully delegated paint node rendering.
The most basic implementation of a ClutterEffect.paint_node
vfunc, and also the default implementation, is with an actor
node, as follows:
```
static void
foo_bar_paint_node (ClutterEffect *effect,
ClutterPaintNode *node,
ClutterPaintContext *paint_context,
ClutterEffectPaintFlags flags)
{
g_autoptr (ClutterPaintNode) actor_node = NULL;
actor_node = clutter_actor_node_new (effect->actor);
clutter_paint_node_add_child (node, actor_node);
}
```
This example gives the exact same behavior of simply calling
clutter_actor_continue_paint(). In the future, the paint node
itself will be a parameter of clutter_actor_continue_paint()
and we'll be able to simplify it event more.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1340>
2020-06-27 15:12:03 +00:00
|
|
|
|
|
2011-03-02 16:55:10 +00:00
|
|
|
|
if (pre_paint_succeeded)
|
2020-12-29 17:43:22 +00:00
|
|
|
|
{
|
|
|
|
|
effect_class->paint_node (effect, node, paint_context, flags);
|
|
|
|
|
effect_class->post_paint (effect, node, paint_context);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Just paint the actor as fallback */
|
|
|
|
|
add_actor_node (effect, node);
|
|
|
|
|
}
|
2011-03-02 16:55:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-02 12:16:23 +00:00
|
|
|
|
static void
|
2019-11-20 20:46:41 +00:00
|
|
|
|
clutter_effect_real_pick (ClutterEffect *effect,
|
|
|
|
|
ClutterPickContext *pick_context)
|
2011-06-02 12:16:23 +00:00
|
|
|
|
{
|
|
|
|
|
ClutterActorMeta *actor_meta = CLUTTER_ACTOR_META (effect);
|
|
|
|
|
ClutterActor *actor;
|
|
|
|
|
|
|
|
|
|
actor = clutter_actor_meta_get_actor (actor_meta);
|
2019-11-20 20:46:41 +00:00
|
|
|
|
clutter_actor_continue_pick (actor, pick_context);
|
2011-06-02 12:16:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-01 18:36:55 +00:00
|
|
|
|
static void
|
2020-04-08 15:02:28 +00:00
|
|
|
|
clutter_effect_set_enabled (ClutterActorMeta *meta,
|
|
|
|
|
gboolean is_enabled)
|
2011-03-01 18:36:55 +00:00
|
|
|
|
{
|
2020-04-08 15:02:28 +00:00
|
|
|
|
ClutterActorMetaClass *parent_class =
|
|
|
|
|
CLUTTER_ACTOR_META_CLASS (clutter_effect_parent_class);
|
|
|
|
|
ClutterActor *actor;
|
2011-03-01 18:36:55 +00:00
|
|
|
|
|
2020-04-08 15:02:28 +00:00
|
|
|
|
actor = clutter_actor_meta_get_actor (meta);
|
|
|
|
|
if (actor)
|
|
|
|
|
clutter_actor_queue_redraw (actor);
|
2011-03-01 18:36:55 +00:00
|
|
|
|
|
2020-04-08 15:02:28 +00:00
|
|
|
|
parent_class->set_enabled (meta, is_enabled);
|
2011-03-01 18:36:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-04-08 09:55:15 +00:00
|
|
|
|
static void
|
|
|
|
|
clutter_effect_class_init (ClutterEffectClass *klass)
|
|
|
|
|
{
|
2020-04-08 15:02:28 +00:00
|
|
|
|
ClutterActorMetaClass *actor_meta_class = CLUTTER_ACTOR_META_CLASS (klass);
|
2011-03-01 18:36:55 +00:00
|
|
|
|
|
2020-04-08 15:02:28 +00:00
|
|
|
|
actor_meta_class->set_enabled = clutter_effect_set_enabled;
|
2011-03-01 18:36:55 +00:00
|
|
|
|
|
2010-04-08 09:55:15 +00:00
|
|
|
|
klass->pre_paint = clutter_effect_real_pre_paint;
|
|
|
|
|
klass->post_paint = clutter_effect_real_post_paint;
|
2018-05-23 06:46:40 +00:00
|
|
|
|
klass->modify_paint_volume = clutter_effect_real_modify_paint_volume;
|
2011-06-06 06:44:11 +00:00
|
|
|
|
klass->paint = clutter_effect_real_paint;
|
clutter/effect: Add paint_node vfunc
Introduce a new paint_node vfunc that, if implemented, allows
the effect to add nodes to a transient paint node that is
immediately painted. This is a transitional step until we
have fully delegated paint node rendering.
The most basic implementation of a ClutterEffect.paint_node
vfunc, and also the default implementation, is with an actor
node, as follows:
```
static void
foo_bar_paint_node (ClutterEffect *effect,
ClutterPaintNode *node,
ClutterPaintContext *paint_context,
ClutterEffectPaintFlags flags)
{
g_autoptr (ClutterPaintNode) actor_node = NULL;
actor_node = clutter_actor_node_new (effect->actor);
clutter_paint_node_add_child (node, actor_node);
}
```
This example gives the exact same behavior of simply calling
clutter_actor_continue_paint(). In the future, the paint node
itself will be a parameter of clutter_actor_continue_paint()
and we'll be able to simplify it event more.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1340>
2020-06-27 15:12:03 +00:00
|
|
|
|
klass->paint_node = clutter_effect_real_paint_node;
|
2011-06-02 12:16:23 +00:00
|
|
|
|
klass->pick = clutter_effect_real_pick;
|
2010-04-08 09:55:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
clutter_effect_init (ClutterEffect *self)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2011-03-02 16:55:10 +00:00
|
|
|
|
void
|
2011-06-13 14:58:46 +00:00
|
|
|
|
_clutter_effect_paint (ClutterEffect *effect,
|
2020-07-05 22:17:09 +00:00
|
|
|
|
ClutterPaintNode *node,
|
2019-11-13 21:21:58 +00:00
|
|
|
|
ClutterPaintContext *paint_context,
|
2011-06-13 14:58:46 +00:00
|
|
|
|
ClutterEffectPaintFlags flags)
|
2011-03-02 16:55:10 +00:00
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (CLUTTER_IS_EFFECT (effect));
|
|
|
|
|
|
2020-07-05 21:45:03 +00:00
|
|
|
|
CLUTTER_EFFECT_GET_CLASS (effect)->paint (effect,
|
|
|
|
|
node,
|
|
|
|
|
paint_context,
|
|
|
|
|
flags);
|
2011-03-02 16:55:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-02 12:16:23 +00:00
|
|
|
|
void
|
2019-11-20 20:46:41 +00:00
|
|
|
|
_clutter_effect_pick (ClutterEffect *effect,
|
|
|
|
|
ClutterPickContext *pick_context)
|
2011-06-02 12:16:23 +00:00
|
|
|
|
{
|
|
|
|
|
g_return_if_fail (CLUTTER_IS_EFFECT (effect));
|
|
|
|
|
|
2019-11-20 20:46:41 +00:00
|
|
|
|
CLUTTER_EFFECT_GET_CLASS (effect)->pick (effect, pick_context);
|
2011-06-02 12:16:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-09-07 17:04:19 +00:00
|
|
|
|
gboolean
|
2018-05-23 06:46:40 +00:00
|
|
|
|
_clutter_effect_modify_paint_volume (ClutterEffect *effect,
|
|
|
|
|
ClutterPaintVolume *volume)
|
2010-08-16 16:02:15 +00:00
|
|
|
|
{
|
2010-09-07 17:04:19 +00:00
|
|
|
|
g_return_val_if_fail (CLUTTER_IS_EFFECT (effect), FALSE);
|
|
|
|
|
g_return_val_if_fail (volume != NULL, FALSE);
|
2010-08-16 16:02:15 +00:00
|
|
|
|
|
2018-05-23 06:46:40 +00:00
|
|
|
|
return CLUTTER_EFFECT_GET_CLASS (effect)->modify_paint_volume (effect,
|
|
|
|
|
volume);
|
2010-08-16 16:02:15 +00:00
|
|
|
|
}
|
2011-02-28 12:16:55 +00:00
|
|
|
|
|
2018-08-29 00:32:29 +00:00
|
|
|
|
gboolean
|
|
|
|
|
_clutter_effect_has_custom_paint_volume (ClutterEffect *effect)
|
|
|
|
|
{
|
|
|
|
|
g_return_val_if_fail (CLUTTER_IS_EFFECT (effect), FALSE);
|
|
|
|
|
|
2018-05-23 06:46:40 +00:00
|
|
|
|
return CLUTTER_EFFECT_GET_CLASS (effect)->modify_paint_volume != clutter_effect_real_modify_paint_volume;
|
2018-08-29 00:32:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-02-28 12:16:55 +00:00
|
|
|
|
/**
|
2011-06-06 06:44:11 +00:00
|
|
|
|
* clutter_effect_queue_repaint:
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* @effect: A #ClutterEffect which needs redrawing
|
|
|
|
|
*
|
2011-06-06 06:44:11 +00:00
|
|
|
|
* Queues a repaint of the effect. The effect can detect when the ‘paint’
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* method is called as a result of this function because it will not
|
2011-09-12 12:12:14 +00:00
|
|
|
|
* have the %CLUTTER_EFFECT_PAINT_ACTOR_DIRTY flag set. In that case the
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* effect is free to assume that the actor has not changed its
|
|
|
|
|
* appearance since the last time it was painted so it doesn't need to
|
|
|
|
|
* call clutter_actor_continue_paint() if it can draw a cached
|
|
|
|
|
* image. This is mostly intended for effects that are using a
|
|
|
|
|
* %CoglOffscreen to redirect the actor (such as
|
|
|
|
|
* %ClutterOffscreenEffect). In that case the effect can save a bit of
|
|
|
|
|
* rendering time by painting the cached texture without causing the
|
|
|
|
|
* entire actor to be painted.
|
|
|
|
|
*
|
|
|
|
|
* This function can be used by effects that have their own animatable
|
|
|
|
|
* parameters. For example, an effect which adds a varying degree of a
|
|
|
|
|
* red tint to an actor by redirecting it through a CoglOffscreen
|
|
|
|
|
* might have a property to specify the level of tint. When this value
|
|
|
|
|
* changes, the underlying actor doesn't need to be redrawn so the
|
2011-06-06 06:44:11 +00:00
|
|
|
|
* effect can call clutter_effect_queue_repaint() to make sure the
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* effect is repainted.
|
|
|
|
|
*
|
|
|
|
|
* Note however that modifying the position of the parent of an actor
|
|
|
|
|
* may change the appearance of the actor because its transformation
|
|
|
|
|
* matrix would change. In this case a redraw wouldn't be queued on
|
2011-09-12 12:12:14 +00:00
|
|
|
|
* the actor itself so the %CLUTTER_EFFECT_PAINT_ACTOR_DIRTY would still
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* not be set. The effect can detect this case by keeping track of the
|
|
|
|
|
* last modelview matrix that was used to render the actor and
|
2020-08-26 09:49:50 +00:00
|
|
|
|
* verifying that it remains the same in the next paint.
|
2011-02-28 12:16:55 +00:00
|
|
|
|
*
|
|
|
|
|
* Any other effects that are layered on top of the passed in effect
|
2011-09-12 12:12:14 +00:00
|
|
|
|
* will still be passed the %CLUTTER_EFFECT_PAINT_ACTOR_DIRTY flag. If
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* anything queues a redraw on the actor without specifying an effect
|
|
|
|
|
* or with an effect that is lower in the chain of effects than this
|
|
|
|
|
* one then that will override this call. In that case this effect
|
2011-09-12 12:12:14 +00:00
|
|
|
|
* will instead be called with the %CLUTTER_EFFECT_PAINT_ACTOR_DIRTY
|
2011-02-28 12:16:55 +00:00
|
|
|
|
* flag set.
|
|
|
|
|
*/
|
|
|
|
|
void
|
2011-06-06 06:44:11 +00:00
|
|
|
|
clutter_effect_queue_repaint (ClutterEffect *effect)
|
2011-02-28 12:16:55 +00:00
|
|
|
|
{
|
|
|
|
|
ClutterActor *actor;
|
|
|
|
|
|
|
|
|
|
g_return_if_fail (CLUTTER_IS_EFFECT (effect));
|
|
|
|
|
|
|
|
|
|
actor = clutter_actor_meta_get_actor (CLUTTER_ACTOR_META (effect));
|
|
|
|
|
|
|
|
|
|
/* If the effect has no actor then nothing needs to be done */
|
|
|
|
|
if (actor != NULL)
|
|
|
|
|
_clutter_actor_queue_redraw_full (actor,
|
|
|
|
|
NULL, /* clip volume */
|
|
|
|
|
effect /* effect */);
|
|
|
|
|
}
|