2007-05-10 Matthew Allum <mallum@openedhand.com>

* clutter/clutter-backend.c:
        * clutter/clutter-backend.h:
        * clutter/clutter-event.c:
        * clutter/clutter-main.c:
        * clutter/clutter-private.h:
        * clutter/glx/clutter-backend-glx.c:
        * clutter/glx/clutter-backend-glx.h:
        * clutter/glx/clutter-event-glx.c:
        Rejig backend event code as to simplify a little.
        NOTE: This breaks non glx backends.

        * tests/Makefile.am:
        * tests/test-events.c:
        Add a very simple event test.

        * tests/test-textures.c: (main):
        Add some more sizes to the test.
This commit is contained in:
Matthew Allum 2007-05-09 23:31:08 +00:00
parent e1bf19f869
commit eb42f82f23
12 changed files with 306 additions and 508 deletions

View File

@ -1,3 +1,23 @@
2007-05-10 Matthew Allum <mallum@openedhand.com>
* clutter/clutter-backend.c:
* clutter/clutter-backend.h:
* clutter/clutter-event.c:
* clutter/clutter-main.c:
* clutter/clutter-private.h:
* clutter/glx/clutter-backend-glx.c:
* clutter/glx/clutter-backend-glx.h:
* clutter/glx/clutter-event-glx.c:
Rejig backend event code as to simplify a little.
NOTE: This breaks non glx backends.
* tests/Makefile.am:
* tests/test-events.c:
Add a very simple event test.
* tests/test-textures.c: (main):
Add some more sizes to the test.
2007-05-05 Matthew Allum <mallum@openedhand.com> 2007-05-05 Matthew Allum <mallum@openedhand.com>
* clutter/clutter-texture.c: * clutter/clutter-texture.c:

View File

@ -34,16 +34,30 @@ G_DEFINE_ABSTRACT_TYPE (ClutterBackend,
clutter_backend, clutter_backend,
G_TYPE_OBJECT); G_TYPE_OBJECT);
#define CLUTTER_BACKEND_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE ((obj), CLUTTER_TYPE_BACKEND, ClutterBackendPrivate))
struct _ClutterBackendPrivate
{
/* settings */
guint double_click_time;
guint double_click_distance;
};
static void static void
clutter_backend_dispose (GObject *gobject) clutter_backend_dispose (GObject *gobject)
{ {
ClutterBackend *backend = CLUTTER_BACKEND (gobject); ClutterMainContext *clutter_context;
if (backend->events_queue) clutter_context = clutter_context_get_default ();
if (clutter_context && clutter_context->events_queue)
{ {
g_queue_foreach (backend->events_queue, (GFunc) clutter_event_free, NULL); g_queue_foreach (clutter_context->events_queue, (GFunc) clutter_event_free, NULL);
g_queue_free (backend->events_queue); g_queue_free (clutter_context->events_queue);
backend->events_queue = NULL; clutter_context->events_queue = NULL;
} }
G_OBJECT_CLASS (clutter_backend_parent_class)->dispose (gobject); G_OBJECT_CLASS (clutter_backend_parent_class)->dispose (gobject);
@ -55,12 +69,16 @@ clutter_backend_class_init (ClutterBackendClass *klass)
GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->dispose = clutter_backend_dispose; gobject_class->dispose = clutter_backend_dispose;
g_type_class_add_private (gobject_class, sizeof (ClutterBackendPrivate));
} }
static void static void
clutter_backend_init (ClutterBackend *backend) clutter_backend_init (ClutterBackend *backend)
{ {
ClutterBackendPrivate *priv;
priv = backend->priv = CLUTTER_BACKEND_GET_PRIVATE(backend);
} }
ClutterActor * ClutterActor *
@ -125,18 +143,6 @@ _clutter_backend_init_stage (ClutterBackend *backend,
return TRUE; return TRUE;
} }
void
_clutter_backend_init_events (ClutterBackend *backend)
{
ClutterBackendClass *klass;
g_return_if_fail (CLUTTER_IS_BACKEND (backend));
klass = CLUTTER_BACKEND_GET_CLASS (backend);
if (klass->init_events)
klass->init_events (backend);
}
void void
_clutter_backend_init_features (ClutterBackend *backend) _clutter_backend_init_features (ClutterBackend *backend)
{ {
@ -149,14 +155,34 @@ _clutter_backend_init_features (ClutterBackend *backend)
klass->init_features (backend); klass->init_features (backend);
} }
void
_clutter_backend_init_events (ClutterBackend *backend)
{
ClutterBackendClass *klass;
ClutterMainContext *clutter_context;
clutter_context = clutter_context_get_default ();
g_return_if_fail (CLUTTER_IS_BACKEND (backend));
g_return_if_fail (clutter_context != NULL);
clutter_context->events_queue = g_queue_new ();
klass = CLUTTER_BACKEND_GET_CLASS (backend);
if (klass->init_events)
klass->init_events (backend);
}
/** /**
* clutter_get_default_backend: * clutter_get_default_backend:
* *
* FIXME * FIXME
* *
* Return value: the default backend. You should not ref or * Return value: the default backend. You should not ref or
* unref the returned object * unref the returned object. Applications should not rarely need
* * to use this.
*
* Since: 0.4 * Since: 0.4
*/ */
ClutterBackend * ClutterBackend *
@ -169,188 +195,7 @@ clutter_get_default_backend (void)
return clutter_context->backend; return clutter_context->backend;
} }
/** /* FIXME: below should probably be moved into clutter_main */
* clutter_backend_get_event:
* @backend: a #ClutterBackend
*
* FIXME
*
* Return value: the #ClutterEvent removed from the queue
*
* Since: 0.4
*/
ClutterEvent *
clutter_backend_get_event (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), NULL);
_clutter_events_queue (backend);
return _clutter_event_queue_pop (backend);
}
/**
* clutter_backend_peek_event:
* @backend: a #ClutterBackend
*
* FIXME
*
* Return value: a copy of the first #ClutterEvent in the queue
*
* Since: 0.4
*/
ClutterEvent *
clutter_backend_peek_event (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), NULL);
return _clutter_event_queue_peek (backend);
}
/**
* clutter_backend_put_event:
* @backend: a #ClutterBackend
* @event: a #ClutterEvent
*
* FIXME
*
* Since: 0.4
*/
void
clutter_backend_put_event (ClutterBackend *backend,
ClutterEvent *event)
{
g_return_if_fail (CLUTTER_IS_BACKEND (backend));
g_return_if_fail (event != NULL);
_clutter_event_queue_push (backend, clutter_event_copy (event));
}
/**
* clutter_backend_get_width:
* @backend: a #ClutterBackend
*
* Gets the width of the screen used by @backend in pixels.
*
* Return value: the width of the screen
*
* Since: 0.4
*/
gint
clutter_backend_get_width (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), -1);
return backend->res_width;
}
/**
* clutter_backend_get_height:
* @backend: a #ClutterBackend
*
* Gets the height of the screen used by @backend in pixels.
*
* Return value: the height of the screen
*
* Since: 0.4
*/
gint
clutter_backend_get_height (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), -1);
return backend->res_height;
}
/**
* clutter_backend_get_width_mm:
* @backend: a #ClutterBackend
*
* Gets the width of the screen used by @backend in millimiters.
*
* Return value: the width of the screen
*
* Since: 0.4
*/
gint
clutter_backend_get_width_mm (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), -1);
return backend->mm_width;
}
/**
* clutter_backend_get_height_mm:
* @backend: a #ClutterBackend
*
* Gets the height of the screen used by @backend in millimiters.
*
* Return value: the height of the screen
*
* Since: 0.4
*/
gint
clutter_backend_get_height_mm (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), -1);
return backend->mm_height;
}
/**
* clutter_backend_get_screen_number:
* @backend: a #ClutterBackend
*
* Gets the number of screens available for @backend.
*
* Return value: the number of screens.
*
* Since: 0.4
*/
gint
clutter_backend_get_screen_number (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), -1);
return backend->screen_num;
}
/**
* clutter_backend_get_n_screens:
* @backend: a #ClutterBackend
*
* Gets the number of screens managed by @backend.
*
* Return value: the number of screens
*
* Since: 0.4
*/
gint
clutter_backend_get_n_screens (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0);
return backend->n_screens;
}
/**
* clutter_backend_get_resolution:
* @backend: a #ClutterBackend
*
* Gets the resolution of the screen used by @backend.
*
* Return value: the resolution of the screen
*
* Since: 0.4
*/
gdouble
clutter_backend_get_resolution (ClutterBackend *backend)
{
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0.0);
return (((gdouble) backend->res_height * 25.4) /
(gdouble) backend->mm_height);
}
/** /**
* clutter_backend_set_double_click_time: * clutter_backend_set_double_click_time:
@ -368,7 +213,7 @@ clutter_backend_set_double_click_time (ClutterBackend *backend,
{ {
g_return_if_fail (CLUTTER_IS_BACKEND (backend)); g_return_if_fail (CLUTTER_IS_BACKEND (backend));
backend->double_click_time = msec; backend->priv->double_click_time = msec;
} }
/** /**
@ -387,7 +232,7 @@ clutter_backend_get_double_click_time (ClutterBackend *backend)
{ {
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0); g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0);
return backend->double_click_time; return backend->priv->double_click_time;
} }
/** /**
@ -405,7 +250,7 @@ clutter_backend_set_double_click_distance (ClutterBackend *backend,
{ {
g_return_if_fail (CLUTTER_IS_BACKEND (backend)); g_return_if_fail (CLUTTER_IS_BACKEND (backend));
backend->double_click_distance = distance; backend->priv->double_click_distance = distance;
} }
/** /**
@ -423,5 +268,5 @@ clutter_backend_get_double_click_distance (ClutterBackend *backend)
{ {
g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0); g_return_val_if_fail (CLUTTER_IS_BACKEND (backend), 0);
return backend->double_click_distance; return backend->priv->double_click_distance;
} }

View File

@ -40,34 +40,14 @@ G_BEGIN_DECLS
#define CLUTTER_BACKEND_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), CLUTTER_TYPE_BACKEND, ClutterBackendClass)) #define CLUTTER_BACKEND_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), CLUTTER_TYPE_BACKEND, ClutterBackendClass))
typedef struct _ClutterBackend ClutterBackend; typedef struct _ClutterBackend ClutterBackend;
typedef struct _ClutterBackendPrivate ClutterBackendPrivate;
typedef struct _ClutterBackendClass ClutterBackendClass; typedef struct _ClutterBackendClass ClutterBackendClass;
struct _ClutterBackend struct _ClutterBackend
{ {
GObject parent_instance; GObject parent_instance;
ClutterBackendPrivate *priv;
/*< private >*/
gint mm_width;
gint mm_height;
gint res_width;
gint res_height;
gint screen_num;
gint n_screens;
/* events queue: every backend must implement one */
GQueue *events_queue;
gpointer queue_head;
/* settings */
guint double_click_time;
guint double_click_distance;
/* multiple button click detection */
guint32 button_click_time[2];
guint32 button_number[2];
gint button_x[2];
gint button_y[2];
}; };
struct _ClutterBackendClass struct _ClutterBackendClass
@ -92,17 +72,12 @@ GType clutter_backend_get_type (void) G_GNUC_CONST;
ClutterBackend *clutter_get_default_backend (void); ClutterBackend *clutter_get_default_backend (void);
#if 0
ClutterEvent *clutter_backend_get_event (ClutterBackend *backend); ClutterEvent *clutter_backend_get_event (ClutterBackend *backend);
ClutterEvent *clutter_backend_peek_event (ClutterBackend *backend); ClutterEvent *clutter_backend_peek_event (ClutterBackend *backend);
void clutter_backend_put_event (ClutterBackend *backend, void clutter_backend_put_event (ClutterBackend *backend,
ClutterEvent *event); ClutterEvent *event);
#endif
gint clutter_backend_get_width (ClutterBackend *backend);
gint clutter_backend_get_height (ClutterBackend *backend);
gint clutter_backend_get_width_mm (ClutterBackend *backend);
gint clutter_backend_get_height_mm (ClutterBackend *backend);
gint clutter_backend_get_screen_number (ClutterBackend *backend);
gint clutter_backend_get_n_screens (ClutterBackend *backend);
void clutter_backend_set_double_click_time (ClutterBackend *backend, void clutter_backend_set_double_click_time (ClutterBackend *backend,
guint msec); guint msec);

View File

@ -32,10 +32,11 @@
#include "clutter-private.h" #include "clutter-private.h"
#include "clutter-debug.h" #include "clutter-debug.h"
/* main event handler */ /* multiple button click detection */
ClutterEventFunc _clutter_event_func = NULL; static guint32 button_click_time[2] = {0, 0};
gpointer _clutter_event_data = NULL; static guint32 button_number[2] = {0, -1};
GDestroyNotify _clutter_event_destroy = NULL; static gint button_x[2] = {0, 0};
static gint button_y[2] = {0, 0};;
/** /**
* clutter_event_type: * clutter_event_type:
@ -320,6 +321,7 @@ clutter_event_new (ClutterEventType type)
new_event = g_slice_new0 (ClutterEvent); new_event = g_slice_new0 (ClutterEvent);
new_event->type = new_event->any.type = type; new_event->type = new_event->any.type = type;
/* FIXME: why do we put in a hash ? */
g_hash_table_insert (event_hash, new_event, GUINT_TO_POINTER (1)); g_hash_table_insert (event_hash, new_event, GUINT_TO_POINTER (1));
return new_event; return new_event;
@ -375,10 +377,8 @@ ClutterEvent *
clutter_event_get (void) clutter_event_get (void)
{ {
ClutterMainContext *context = clutter_context_get_default (); ClutterMainContext *context = clutter_context_get_default ();
ClutterBackend *backend = context->backend;
_clutter_events_queue (backend); return g_queue_pop_tail (context->events_queue);
return _clutter_event_queue_pop (backend);
} }
/** /**
@ -394,8 +394,13 @@ ClutterEvent *
clutter_event_peek (void) clutter_event_peek (void)
{ {
ClutterMainContext *context = clutter_context_get_default (); ClutterMainContext *context = clutter_context_get_default ();
g_return_val_if_fail (context != NULL, NULL);
return _clutter_event_queue_peek (context->backend); if (context->events_queue == NULL)
return NULL;
return g_queue_peek_tail (context->events_queue);
} }
/** /**
@ -410,74 +415,32 @@ void
clutter_event_put (ClutterEvent *event) clutter_event_put (ClutterEvent *event)
{ {
ClutterMainContext *context = clutter_context_get_default (); ClutterMainContext *context = clutter_context_get_default ();
ClutterBackend *backend = context->backend;
_clutter_event_queue_push (backend, clutter_event_copy (event)); /* FIXME: check queue is valid */
} g_return_if_fail (context != NULL);
void g_queue_push_head (context->events_queue, clutter_event_copy (event));
_clutter_event_queue_push (ClutterBackend *backend,
ClutterEvent *event)
{
if (!backend->events_queue)
backend->events_queue = g_queue_new ();
g_queue_push_head (backend->events_queue, event);
}
ClutterEvent *
_clutter_event_queue_pop (ClutterBackend *backend)
{
if (!backend->events_queue)
return NULL;
return g_queue_pop_tail (backend->events_queue);
}
ClutterEvent *
_clutter_event_queue_peek (ClutterBackend *backend)
{
if (!backend->events_queue)
return NULL;
return g_queue_peek_tail (backend->events_queue);
} }
gboolean gboolean
_clutter_event_queue_check_pending (ClutterBackend *backend) clutter_events_pending (void)
{ {
if (!backend->events_queue) ClutterMainContext *context = clutter_context_get_default ();
g_return_val_if_fail (context != NULL, FALSE);
if (!context->events_queue)
return FALSE; return FALSE;
return g_queue_is_empty (backend->events_queue) == FALSE; return g_queue_is_empty (context->events_queue) == FALSE;
} }
void /* Backend helpers (private) */
_clutter_set_events_handler (ClutterEventFunc func,
gpointer data,
GDestroyNotify destroy)
{
if (_clutter_event_destroy)
(* _clutter_event_destroy) (_clutter_event_data);
_clutter_event_func = func; static void
_clutter_event_data = data; synthesize_click (ClutterBackend *backend,
_clutter_event_destroy = destroy; ClutterEvent *event,
} gint n_clicks)
void
_clutter_synthetise_stage_state (ClutterBackend *backend,
ClutterEvent *event,
ClutterStageState set_flags,
ClutterStageState unset_flags)
{
}
void
_clutter_synthesize_click (ClutterBackend *backend,
ClutterEvent *event,
gint n_clicks)
{ {
ClutterEvent temp_event; ClutterEvent temp_event;
@ -485,52 +448,58 @@ _clutter_synthesize_click (ClutterBackend *backend,
temp_event.type = (n_clicks == 2) ? CLUTTER_2BUTTON_PRESS temp_event.type = (n_clicks == 2) ? CLUTTER_2BUTTON_PRESS
: CLUTTER_3BUTTON_PRESS; : CLUTTER_3BUTTON_PRESS;
clutter_backend_put_event (backend, &temp_event); clutter_event_put (&temp_event);
} }
/* post process a button to synthesize double clicks etc */
void void
_clutter_event_button_generate (ClutterBackend *backend, _clutter_event_button_generate (ClutterBackend *backend,
ClutterEvent *event) ClutterEvent *event)
{ {
if ((event->button.time < (backend->button_click_time[1] + 2 * backend->double_click_time)) && guint double_click_time, double_click_distance;
(event->button.button == backend->button_number[1]) &&
(ABS (event->button.x - backend->button_x[1]) <= backend->double_click_distance) && double_click_distance = clutter_backend_get_double_click_distance (backend);
(ABS (event->button.y - backend->button_y[1]) <= backend->double_click_distance)) double_click_time = clutter_backend_get_double_click_time (backend);
if ((event->button.time < (button_click_time[1] + 2 * double_click_time))
&& (event->button.button == button_number[1])
&& (ABS (event->button.x - button_x[1]) <= double_click_distance)
&& (ABS (event->button.y - button_y[1]) <= double_click_distance))
{ {
_clutter_synthesize_click (backend, event, 3); synthesize_click (backend, event, 3);
backend->button_click_time[1] = 0; button_click_time[1] = 0;
backend->button_click_time[0] = 0; button_click_time[0] = 0;
backend->button_number[1] = -1; button_number[1] = -1;
backend->button_number[0] = -1; button_number[0] = -1;
backend->button_x[0] = backend->button_x[1] = 0; button_x[0] = button_x[1] = 0;
backend->button_y[0] = backend->button_y[1] = 0; button_y[0] = button_y[1] = 0;
} }
else if ((event->button.time < (backend->button_click_time[0] + backend->double_click_time)) && else if ((event->button.time < (button_click_time[0] + double_click_time)) &&
(event->button.button == backend->button_number[0]) && (event->button.button == button_number[0]) &&
(ABS (event->button.x - backend->button_x[0]) <= backend->double_click_distance) && (ABS (event->button.x - button_x[0]) <= double_click_distance) &&
(ABS (event->button.y - backend->button_y[0]) <= backend->double_click_distance)) (ABS (event->button.y - button_y[0]) <= double_click_distance))
{ {
_clutter_synthesize_click (backend, event, 2); synthesize_click (backend, event, 2);
backend->button_click_time[1] = backend->button_click_time[0]; button_click_time[1] = button_click_time[0];
backend->button_click_time[0] = event->button.time; button_click_time[0] = event->button.time;
backend->button_number[1] = backend->button_number[0]; button_number[1] = button_number[0];
backend->button_number[0] = event->button.button; button_number[0] = event->button.button;
backend->button_x[1] = backend->button_x[0]; button_x[1] = button_x[0];
backend->button_x[0] = event->button.x; button_x[0] = event->button.x;
backend->button_y[1] = backend->button_y[0]; button_y[1] = button_y[0];
backend->button_y[0] = event->button.y; button_y[0] = event->button.y;
} }
else else
{ {
backend->button_click_time[1] = 0; button_click_time[1] = 0;
backend->button_click_time[0] = event->button.time; button_click_time[0] = event->button.time;
backend->button_number[1] = -1; button_number[1] = -1;
backend->button_number[0] = event->button.button; button_number[0] = event->button.button;
backend->button_x[1] = 0; button_x[1] = 0;
backend->button_x[0] = event->button.x; button_x[0] = event->button.x;
backend->button_y[1] = 0; button_y[1] = 0;
backend->button_y[0] = event->button.y; button_y[0] = event->button.y;
} }
} }

View File

@ -70,7 +70,6 @@ static const GDebugKey clutter_debug_keys[] = {
}; };
#endif /* CLUTTER_ENABLE_DEBUG */ #endif /* CLUTTER_ENABLE_DEBUG */
/** /**
* clutter_get_show_fps: * clutter_get_show_fps:
* *
@ -116,9 +115,13 @@ clutter_redraw (void)
clutter_actor_paint (_clutter_backend_get_stage (ctx->backend)); clutter_actor_paint (_clutter_backend_get_stage (ctx->backend));
} }
static void /**
clutter_main_do_event (ClutterEvent *event, * clutter_do_event
gpointer dummy) *
* This function should never be called by applications.
*/
void
clutter_do_event (ClutterEvent *event)
{ {
ClutterMainContext *context; ClutterMainContext *context;
ClutterBackend *backend; ClutterBackend *backend;
@ -367,8 +370,6 @@ pre_parse_hook (GOptionContext *context,
backend = clutter_context->backend; backend = clutter_context->backend;
g_assert (CLUTTER_IS_BACKEND (backend)); g_assert (CLUTTER_IS_BACKEND (backend));
_clutter_set_events_handler (clutter_main_do_event, NULL, NULL);
#ifdef CLUTTER_ENABLE_DEBUG #ifdef CLUTTER_ENABLE_DEBUG
env_string = g_getenv ("CLUTTER_DEBUG"); env_string = g_getenv ("CLUTTER_DEBUG");
if (env_string != NULL) if (env_string != NULL)
@ -631,7 +632,7 @@ clutter_init (int *argc,
return CLUTTER_INIT_ERROR_INTERNAL; return CLUTTER_INIT_ERROR_INTERNAL;
} }
_clutter_events_init (context->backend); _clutter_backend_init_events (context->backend);
return CLUTTER_INIT_SUCCESS; return CLUTTER_INIT_SUCCESS;
} }

View File

@ -47,17 +47,14 @@ typedef struct _ClutterMainContext ClutterMainContext;
struct _ClutterMainContext struct _ClutterMainContext
{ {
/* holds a pointer to the stage */ /* holds a pointer to the stage */
ClutterBackend *backend; ClutterBackend *backend;
GQueue *events_queue;
PangoFT2FontMap *font_map; PangoFT2FontMap *font_map;
GMutex *gl_lock;
GMutex *gl_lock; guint update_idle;
guint update_idle; guint main_loop_level;
GSList *main_loops;
guint main_loop_level; guint is_initialized : 1;
GSList *main_loops;
guint is_initialized : 1;
}; };
#define CLUTTER_CONTEXT() (clutter_context_get_default ()) #define CLUTTER_CONTEXT() (clutter_context_get_default ())
@ -84,19 +81,18 @@ typedef enum {
#define CLUTTER_PARAM_READWRITE \ #define CLUTTER_PARAM_READWRITE \
G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK |G_PARAM_STATIC_BLURB G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK |G_PARAM_STATIC_BLURB
/* signal accumulators */
gboolean _clutter_boolean_accumulator (GSignalInvocationHint *ihint,
GValue *return_accu,
const GValue *handler_return,
gpointer dummy);
/* vfuncs implemnted by backend */
GType _clutter_backend_impl_get_type (void); GType _clutter_backend_impl_get_type (void);
/* backend-specific private functions */
void _clutter_events_init (ClutterBackend *backend);
void _clutter_events_uninit (ClutterBackend *backend);
void _clutter_events_queue (ClutterBackend *backend);
void _clutter_event_queue_push (ClutterBackend *backend,
ClutterEvent *event);
ClutterEvent *_clutter_event_queue_pop (ClutterBackend *backend);
ClutterEvent *_clutter_event_queue_peek (ClutterBackend *backend);
gboolean _clutter_event_queue_check_pending (ClutterBackend *backend);
ClutterActor *_clutter_backend_get_stage (ClutterBackend *backend); ClutterActor *_clutter_backend_get_stage (ClutterBackend *backend);
void _clutter_backend_add_options (ClutterBackend *backend, void _clutter_backend_add_options (ClutterBackend *backend,
GOptionGroup *group); GOptionGroup *group);
gboolean _clutter_backend_pre_parse (ClutterBackend *backend, gboolean _clutter_backend_pre_parse (ClutterBackend *backend,
@ -106,39 +102,19 @@ gboolean _clutter_backend_post_parse (ClutterBackend *backend,
gboolean _clutter_backend_init_stage (ClutterBackend *backend, gboolean _clutter_backend_init_stage (ClutterBackend *backend,
GError **error); GError **error);
void _clutter_backend_init_events (ClutterBackend *backend); void _clutter_backend_init_events (ClutterBackend *backend);
void _clutter_backend_init_features (ClutterBackend *backend); void _clutter_backend_init_features (ClutterBackend *backend);
/* backend helpers */
/* signal accumulators */
gboolean _clutter_boolean_accumulator (GSignalInvocationHint *ihint,
GValue *return_accu,
const GValue *handler_return,
gpointer dummy);
typedef void (* ClutterEventFunc) (ClutterEvent *event,
gpointer data);
/* the event dispatcher function */
extern ClutterEventFunc _clutter_event_func;
extern gpointer _clutter_event_data;
extern GDestroyNotify _clutter_event_destroy;
void _clutter_set_events_handler (ClutterEventFunc func,
gpointer data,
GDestroyNotify destroy);
void _clutter_event_button_generate (ClutterBackend *backend, void _clutter_event_button_generate (ClutterBackend *backend,
ClutterEvent *event); ClutterEvent *event);
void _clutter_synthetise_click (ClutterBackend *backend,
ClutterEvent *event,
gint n_clicks);
void _clutter_synthetise_stage_state (ClutterBackend *backend,
ClutterEvent *event,
ClutterStageState set_flags,
ClutterStageState unset_flags);
/* FIXME: move elsewhere via ClutterAudience */
void _clutter_stage_sync_viewport (ClutterStage *stage); void _clutter_stage_sync_viewport (ClutterStage *stage);
/* Does this need to be private ? */
void clutter_do_event (ClutterEvent *event);
G_END_DECLS G_END_DECLS
#endif /* _HAVE_CLUTTER_PRIVATE_H */ #endif /* _HAVE_CLUTTER_PRIVATE_H */

View File

@ -107,13 +107,6 @@ clutter_backend_glx_post_parse (ClutterBackend *backend,
backend_glx->display_name = g_strdup (clutter_display_name); backend_glx->display_name = g_strdup (clutter_display_name);
/* generic backend properties */
backend->res_width = WidthOfScreen (backend_glx->xscreen);
backend->res_height = HeightOfScreen (backend_glx->xscreen);
backend->mm_width = WidthMMOfScreen (backend_glx->xscreen);
backend->mm_height = HeightMMOfScreen (backend_glx->xscreen);
backend->screen_num = backend_glx->xscreen_num;
backend->n_screens = ScreenCount (backend_glx->xdpy);
} }
g_free (clutter_display_name); g_free (clutter_display_name);
@ -130,7 +123,7 @@ clutter_backend_glx_post_parse (ClutterBackend *backend,
static gboolean static gboolean
is_gl_version_at_least_12 (void) is_gl_version_at_least_12 (void)
{ {
/* FIXME: This likely needs to live elsewhere in features */ /* FIXME: This likely needs to live elsewhere in features or cogl */
return return
(g_ascii_strtod ((const gchar*) glGetString (GL_VERSION), NULL) >= 1.2); (g_ascii_strtod ((const gchar*) glGetString (GL_VERSION), NULL) >= 1.2);
} }
@ -189,7 +182,8 @@ static void
clutter_backend_glx_init_events (ClutterBackend *backend) clutter_backend_glx_init_events (ClutterBackend *backend)
{ {
CLUTTER_NOTE (EVENT, "initialising the event loop"); CLUTTER_NOTE (EVENT, "initialising the event loop");
_clutter_events_init (backend);
_clutter_backend_glx_events_init (backend);
} }
static ClutterActor * static ClutterActor *
@ -244,7 +238,7 @@ clutter_backend_glx_dispose (GObject *gobject)
{ {
ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (gobject); ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (gobject);
_clutter_events_uninit (CLUTTER_BACKEND (backend_glx)); _clutter_backend_glx_events_uninit (CLUTTER_BACKEND(backend_glx));
if (backend_glx->stage) if (backend_glx->stage)
{ {
@ -301,21 +295,10 @@ static void
clutter_backend_glx_init (ClutterBackendGlx *backend_glx) clutter_backend_glx_init (ClutterBackendGlx *backend_glx)
{ {
ClutterBackend *backend = CLUTTER_BACKEND (backend_glx); ClutterBackend *backend = CLUTTER_BACKEND (backend_glx);
backend->events_queue = g_queue_new ();
backend->button_click_time[0] = backend->button_click_time[1] = 0; /* FIXME: get from xsettings */
backend->button_number[0] = backend->button_number[1] = -1; clutter_backend_set_double_click_time (backend, 250);
backend->button_x[0] = backend->button_x[1] = 0; clutter_backend_set_double_click_distance (backend, 5);
backend->button_y[0] = backend->button_y[1] = 0;
backend->res_width = backend->res_height = -1;
backend->mm_width = backend->mm_height = -1;
backend->screen_num = 0;
backend->n_screens = 0;
/* FIXME - find a way to set this stuff from XSettings */
backend->double_click_time = 250;
backend->double_click_distance = 5;
} }
/* every backend must implement this function */ /* every backend must implement this function */

View File

@ -64,6 +64,9 @@ struct _ClutterBackendGlxClass
ClutterBackendClass parent_class; ClutterBackendClass parent_class;
}; };
void _clutter_backend_glx_events_init (ClutterBackend *backend);
void _clutter_backend_glx_events_uninit (ClutterBackend *backend);
GType clutter_backend_glx_get_type (void) G_GNUC_CONST; GType clutter_backend_glx_get_type (void) G_GNUC_CONST;
G_END_DECLS G_END_DECLS

View File

@ -103,7 +103,7 @@ clutter_event_source_new (ClutterBackend *backend)
} }
static gboolean static gboolean
clutter_check_xpending (ClutterBackend *backend) check_xpending (ClutterBackend *backend)
{ {
return XPending (CLUTTER_BACKEND_GLX (backend)->xdpy); return XPending (CLUTTER_BACKEND_GLX (backend)->xdpy);
} }
@ -163,7 +163,7 @@ xembed_set_info (Display *xdisplay,
} }
void void
_clutter_events_init (ClutterBackend *backend) _clutter_backend_glx_events_init (ClutterBackend *backend)
{ {
GSource *source; GSource *source;
ClutterEventSource *event_source; ClutterEventSource *event_source;
@ -195,7 +195,7 @@ _clutter_events_init (ClutterBackend *backend)
} }
void void
_clutter_events_uninit (ClutterBackend *backend) _clutter_backend_glx_events_uninit (ClutterBackend *backend)
{ {
ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (backend); ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (backend);
@ -231,41 +231,6 @@ set_user_time (Display *display,
} }
} }
/**
* clutter_events_pending:
*
* FIXME
*
* Return value: FIXME
*
* Since: 0.4
*/
gboolean
clutter_events_pending (void)
{
GList *i;
for (i = event_sources; i != NULL; i = i->next)
{
ClutterEventSource *source = i->data;
ClutterBackend *backend = source->backend;
if (_clutter_event_queue_check_pending (backend))
return TRUE;
}
for (i = event_sources; i != NULL; i = i->next)
{
ClutterEventSource *source = i->data;
ClutterBackend *backend = source->backend;
if (clutter_check_xpending (backend))
return TRUE;
}
return FALSE;
}
static void static void
translate_key_event (ClutterBackend *backend, translate_key_event (ClutterBackend *backend,
ClutterEvent *event, ClutterEvent *event,
@ -381,9 +346,9 @@ handle_xembed_event (ClutterBackendGlx *backend_glx,
} }
static gboolean static gboolean
clutter_event_translate (ClutterBackend *backend, event_translate (ClutterBackend *backend,
ClutterEvent *event, ClutterEvent *event,
XEvent *xevent) XEvent *xevent)
{ {
ClutterBackendGlx *backend_glx; ClutterBackendGlx *backend_glx;
ClutterStage *stage; ClutterStage *stage;
@ -518,32 +483,27 @@ clutter_event_translate (ClutterBackend *backend,
return res; return res;
} }
void static void
_clutter_events_queue (ClutterBackend *backend) events_queue (ClutterBackend *backend)
{ {
ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (backend); ClutterBackendGlx *backend_glx = CLUTTER_BACKEND_GLX (backend);
ClutterEvent *event; ClutterEvent *event;
XEvent xevent; Display *xdisplay = backend_glx->xdpy;
Display *xdisplay = backend_glx->xdpy; XEvent xevent;
ClutterMainContext *clutter_context;
while (!_clutter_event_queue_check_pending (backend) && XPending (xdisplay)) clutter_context = clutter_context_get_default ();
while (!clutter_events_pending () && XPending (xdisplay))
{ {
XNextEvent (xdisplay, &xevent); XNextEvent (xdisplay, &xevent);
switch (xevent.type)
{
case KeyPress:
case KeyRelease:
break;
default:
if (XFilterEvent (&xevent, None))
continue;
}
event = clutter_event_new (CLUTTER_NOTHING); event = clutter_event_new (CLUTTER_NOTHING);
if (clutter_event_translate (backend, event, &xevent))
if (event_translate (backend, event, &xevent))
{ {
_clutter_event_queue_push (backend, event); /* push directly here to avoid copy of queue_put */
g_queue_push_head (clutter_context->events_queue, event);
} }
else else
{ {
@ -560,8 +520,7 @@ clutter_event_prepare (GSource *source,
gboolean retval; gboolean retval;
*timeout = -1; *timeout = -1;
retval = (_clutter_event_queue_check_pending (backend) || retval = (clutter_events_pending () || check_xpending (backend));
clutter_check_xpending (backend));
return retval; return retval;
} }
@ -574,8 +533,7 @@ clutter_event_check (GSource *source)
gboolean retval; gboolean retval;
if (event_source->event_poll_fd.revents & G_IO_IN) if (event_source->event_poll_fd.revents & G_IO_IN)
retval = (_clutter_event_queue_check_pending (backend) || retval = (clutter_events_pending () || check_xpending (backend));
clutter_check_xpending (backend));
else else
retval = FALSE; retval = FALSE;
@ -590,17 +548,18 @@ clutter_event_dispatch (GSource *source,
ClutterBackend *backend = ((ClutterEventSource *) source)->backend; ClutterBackend *backend = ((ClutterEventSource *) source)->backend;
ClutterEvent *event; ClutterEvent *event;
_clutter_events_queue (backend); /* Grab the event(s), translate and figure out double click.
event = _clutter_event_queue_pop (backend); * The push onto queue (stack) if valid.
*/
events_queue (backend);
/* Pop an event off the queue if any */
event = clutter_event_get ();
if (event) if (event)
{ {
if (_clutter_event_func) /* forward the event into clutter for emission etc. */
{ clutter_do_event (event);
CLUTTER_NOTE (EVENT, "Dispatching _clutter_event_func");
(* _clutter_event_func) (event, _clutter_event_data);
}
clutter_event_free (event); clutter_event_free (event);
} }

View File

@ -1,4 +1,4 @@
noinst_PROGRAMS = test-textures noinst_PROGRAMS = test-textures test-events
INCLUDES = -I$(top_srcdir)/ INCLUDES = -I$(top_srcdir)/
LDADD = $(top_builddir)/clutter/libclutter-@CLUTTER_FLAVOUR@-@CLUTTER_MAJORMINOR@.la LDADD = $(top_builddir)/clutter/libclutter-@CLUTTER_FLAVOUR@-@CLUTTER_MAJORMINOR@.la
@ -6,3 +6,5 @@ AM_CFLAGS = $(CLUTTER_CFLAGS)
LDFLAGS = $(CLUTTER_LIBS) LDFLAGS = $(CLUTTER_LIBS)
test_textures_SOURCES = test-textures.c test_textures_SOURCES = test-textures.c
test_events_SOURCES = test-events.c

63
tests/test-events.c Normal file
View File

@ -0,0 +1,63 @@
#include <clutter/clutter.h>
static void
input_cb (ClutterStage *stage,
ClutterEvent *event,
gpointer data)
{
switch (event->type)
{
case CLUTTER_KEY_PRESS:
printf("- KEY PRESS\n");
break;
case CLUTTER_KEY_RELEASE:
printf("- KEY RELEASE\n");
break;
case CLUTTER_MOTION:
printf("- MOTION\n");
break;
case CLUTTER_BUTTON_PRESS:
printf("- BUTTON PRESS\n");
break;
case CLUTTER_2BUTTON_PRESS:
printf("- BUTTON 2 PRESS\n");
break;
case CLUTTER_3BUTTON_PRESS:
printf("- BUTTON 3 PRESS\n");
break;
case CLUTTER_BUTTON_RELEASE:
printf("- BUTTON RELEASE\n");
break;
case CLUTTER_SCROLL:
printf("- BUTTON SCROLL\n");
break;
case CLUTTER_STAGE_STATE:
printf("- STAGE STATE\n");
break;
case CLUTTER_DESTROY_NOTIFY:
printf("- DESTROY NOTIFY\n");
break;
case CLUTTER_CLIENT_MESSAGE:
printf("- CLIENT MESSAGE\n");
break;
case CLUTTER_DELETE:
printf("- DELETE\n");
break;
}
}
int
main (int argc, char *argv[])
{
ClutterActor *stage;
clutter_init (&argc, &argv);
stage = clutter_stage_get_default ();
g_signal_connect (stage, "event", G_CALLBACK (input_cb), NULL);
clutter_actor_show_all (CLUTTER_ACTOR (stage));
clutter_main();
return 0;
}

View File

@ -56,7 +56,7 @@ main (int argc, char *argv[])
ClutterActor *texture; ClutterActor *texture;
ClutterActor *stage; ClutterActor *stage;
GdkPixbuf *pixbuf; GdkPixbuf *pixbuf;
gint i, cols, rows; gint i, j, cols, rows;
clutter_init (&argc, &argv); clutter_init (&argc, &argv);
@ -66,31 +66,33 @@ main (int argc, char *argv[])
SPIN(); SPIN();
for (i=100; i<5000; i += 100) for (i=100; i<5000; i += 100)
{ for (j=0; j<4; j++)
pixbuf = make_pixbuf (i, i, 4, TRUE); {
pixbuf = make_pixbuf (i+j, i+j, 4, TRUE);
if (!pixbuf)
g_error("%ix%i pixbuf creation failed", i, i); if (!pixbuf)
g_error("%ix%i pixbuf creation failed", i+j, i+j);
printf("o %ix%i pixbuf... ", i, i);
printf("o %ix%i pixbuf... ", i+j, i+j);
texture = clutter_texture_new_from_pixbuf (pixbuf);
texture = clutter_texture_new_from_pixbuf (pixbuf);
g_object_unref (pixbuf);
g_object_unref (pixbuf);
if (!texture)
g_error("Pixbuf creation failed"); if (!texture)
g_error("Pixbuf creation failed");
printf("uploaded to texture... ", i, i);
printf("uploaded to texture... ");
clutter_group_add (CLUTTER_GROUP (stage), texture);
clutter_actor_set_size (texture, 400, 400); clutter_group_add (CLUTTER_GROUP (stage), texture);
clutter_actor_show (texture); clutter_actor_set_size (texture, 400, 400);
clutter_actor_show (texture);
clutter_texture_get_n_tiles(CLUTTER_TEXTURE(texture), &cols, &rows);
printf("with tiles: %i x %i\n", cols, rows); clutter_texture_get_n_tiles(CLUTTER_TEXTURE(texture), &cols, &rows);
SPIN(); printf("with tiles: %i x %i\n", cols, rows);
SPIN();
clutter_group_remove (CLUTTER_GROUP (stage), texture); clutter_group_remove (CLUTTER_GROUP (stage), texture);
} }