2008-03-25 15:42:50 +00:00
|
|
|
/* Clutter.
|
|
|
|
* An OpenGL based 'interactive canvas' library.
|
|
|
|
* Authored By Matthew Allum <mallum@openedhand.com>
|
|
|
|
* Copyright (C) 2006-2007 OpenedHand
|
|
|
|
*
|
|
|
|
* 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
|
2010-03-01 12:56:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*
|
2008-03-25 15:42:50 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "clutter-backend-win32.h"
|
|
|
|
#include "clutter-stage-win32.h"
|
|
|
|
#include "clutter-win32.h"
|
|
|
|
|
2010-10-21 11:16:05 +00:00
|
|
|
#include "clutter-actor-private.h"
|
2010-10-21 10:29:09 +00:00
|
|
|
#include "clutter-main.h"
|
|
|
|
#include "clutter-feature.h"
|
|
|
|
#include "clutter-event.h"
|
|
|
|
#include "clutter-enum-types.h"
|
|
|
|
#include "clutter-private.h"
|
|
|
|
#include "clutter-debug.h"
|
|
|
|
#include "clutter-stage-private.h"
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2008-04-25 13:37:36 +00:00
|
|
|
#include "cogl/cogl.h"
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
2008-04-13 08:43:32 +00:00
|
|
|
static void clutter_stage_window_iface_init (ClutterStageWindowIface *iface);
|
2009-10-26 15:28:36 +00:00
|
|
|
|
2008-04-13 08:43:32 +00:00
|
|
|
G_DEFINE_TYPE_WITH_CODE (ClutterStageWin32,
|
|
|
|
clutter_stage_win32,
|
2009-10-26 15:28:36 +00:00
|
|
|
G_TYPE_OBJECT,
|
2008-04-13 08:43:32 +00:00
|
|
|
G_IMPLEMENT_INTERFACE
|
|
|
|
(CLUTTER_TYPE_STAGE_WINDOW,
|
|
|
|
clutter_stage_window_iface_init));
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
static void
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_stage_win32_show (ClutterStageWindow *stage_window,
|
|
|
|
gboolean do_raise)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2009-10-26 15:28:36 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
if (stage_win32->hwnd)
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
ShowWindow (stage_win32->hwnd, do_raise ? SW_SHOW : SW_SHOWNA);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2010-12-28 18:04:00 +00:00
|
|
|
if (stage_win32->accept_focus)
|
|
|
|
SetForegroundWindow (stage_win32->hwnd);
|
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_actor_map (CLUTTER_ACTOR (stage_win32->wrapper));
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_stage_win32_hide (ClutterStageWindow *stage_window)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2009-10-26 15:28:36 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
2008-06-12 12:12:25 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
if (stage_win32->hwnd)
|
|
|
|
{
|
|
|
|
clutter_actor_unmap (CLUTTER_ACTOR (stage_win32->wrapper));
|
|
|
|
ShowWindow (stage_win32->hwnd, SW_HIDE);
|
|
|
|
}
|
2008-06-12 12:12:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_stage_win32_get_geometry (ClutterStageWindow *stage_window,
|
|
|
|
ClutterGeometry *geometry)
|
2008-06-12 12:12:25 +00:00
|
|
|
{
|
2009-10-26 15:28:36 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
2008-06-12 12:12:25 +00:00
|
|
|
|
|
|
|
if ((stage_win32->state & CLUTTER_STAGE_STATE_FULLSCREEN))
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
geometry->width = (stage_win32->fullscreen_rect.right
|
|
|
|
- stage_win32->fullscreen_rect.left);
|
|
|
|
geometry->height = (stage_win32->fullscreen_rect.bottom
|
|
|
|
- stage_win32->fullscreen_rect.top);
|
|
|
|
return;
|
|
|
|
}
|
2008-06-12 12:12:25 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
geometry->width = stage_win32->win_width;
|
|
|
|
geometry->height = stage_win32->win_height;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_fullscreen_rect (ClutterStageWin32 *stage_win32)
|
|
|
|
{
|
|
|
|
HMONITOR monitor;
|
|
|
|
MONITORINFO monitor_info;
|
|
|
|
|
|
|
|
/* If we already have a window then try to use the same monitor that
|
|
|
|
is already on */
|
|
|
|
if (stage_win32->hwnd)
|
|
|
|
monitor = MonitorFromWindow (stage_win32->hwnd, MONITOR_DEFAULTTONEAREST);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Otherwise just guess that they will want the monitor where
|
|
|
|
the cursor is */
|
|
|
|
POINT cursor;
|
|
|
|
GetCursorPos (&cursor);
|
|
|
|
monitor = MonitorFromPoint (cursor, MONITOR_DEFAULTTONEAREST);
|
|
|
|
}
|
|
|
|
|
|
|
|
monitor_info.cbSize = sizeof (monitor_info);
|
|
|
|
GetMonitorInfoW (monitor, &monitor_info);
|
|
|
|
stage_win32->fullscreen_rect = monitor_info.rcMonitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_full_window_size (ClutterStageWin32 *stage_win32,
|
|
|
|
int width_in, int height_in,
|
|
|
|
int *width_out, int *height_out)
|
|
|
|
{
|
|
|
|
gboolean resizable
|
2008-04-13 08:43:32 +00:00
|
|
|
= clutter_stage_get_user_resizable (stage_win32->wrapper);
|
2008-03-25 15:42:50 +00:00
|
|
|
/* The window size passed to CreateWindow includes the window
|
|
|
|
decorations */
|
|
|
|
*width_out = width_in + GetSystemMetrics (resizable ? SM_CXSIZEFRAME
|
|
|
|
: SM_CXFIXEDFRAME) * 2;
|
|
|
|
*height_out = height_in + GetSystemMetrics (resizable ? SM_CYSIZEFRAME
|
|
|
|
: SM_CYFIXEDFRAME) * 2
|
|
|
|
+ GetSystemMetrics (SM_CYCAPTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_clutter_stage_win32_get_min_max_info (ClutterStageWin32 *stage_win32,
|
|
|
|
MINMAXINFO *min_max_info)
|
|
|
|
{
|
|
|
|
/* If the window isn't resizable then set the max and min size to
|
|
|
|
the current size */
|
2008-04-13 08:43:32 +00:00
|
|
|
if (!clutter_stage_get_user_resizable (CLUTTER_STAGE (stage_win32->wrapper)))
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
|
|
|
int full_width, full_height;
|
|
|
|
get_full_window_size (stage_win32,
|
|
|
|
stage_win32->win_width, stage_win32->win_height,
|
|
|
|
&full_width, &full_height);
|
|
|
|
min_max_info->ptMaxTrackSize.x = full_width;
|
|
|
|
min_max_info->ptMinTrackSize.x = full_width;
|
|
|
|
min_max_info->ptMaxTrackSize.y = full_height;
|
|
|
|
min_max_info->ptMinTrackSize.y = full_height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_stage_win32_resize (ClutterStageWindow *stage_window,
|
|
|
|
gint width,
|
|
|
|
gint height)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2009-10-26 15:28:36 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
gboolean resize;
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
resize = clutter_stage_get_user_resizable (stage_win32->wrapper);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
if (width != stage_win32->win_width || height != stage_win32->win_height)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-06-12 12:12:25 +00:00
|
|
|
/* Ignore size requests if we are in full screen mode */
|
|
|
|
if ((stage_win32->state & CLUTTER_STAGE_STATE_FULLSCREEN) == 0)
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
stage_win32->win_width = width;
|
|
|
|
stage_win32->win_height = height;
|
|
|
|
|
|
|
|
if (stage_win32->hwnd != NULL && !stage_win32->is_foreign_win)
|
|
|
|
{
|
|
|
|
int full_width, full_height;
|
|
|
|
|
|
|
|
get_full_window_size (stage_win32,
|
|
|
|
width, height,
|
|
|
|
&full_width, &full_height);
|
|
|
|
|
|
|
|
SetWindowPos (stage_win32->hwnd, NULL,
|
|
|
|
0, 0,
|
|
|
|
full_width, full_height,
|
|
|
|
SWP_NOZORDER | SWP_NOMOVE);
|
|
|
|
}
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2008-04-13 08:43:32 +00:00
|
|
|
clutter_stage_win32_set_title (ClutterStageWindow *stage_window,
|
|
|
|
const gchar *title)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
/* Empty window titles not allowed, so set it to just a period. */
|
|
|
|
if (title == NULL || !title[0])
|
|
|
|
title = ".";
|
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
if (stage_win32->wtitle != NULL)
|
|
|
|
g_free (stage_win32->wtitle);
|
|
|
|
stage_win32->wtitle = g_utf8_to_utf16 (title, -1, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
/* If the window is not yet created, the title will be set during the
|
|
|
|
window creation */
|
|
|
|
if (stage_win32->hwnd != NULL)
|
|
|
|
SetWindowTextW (stage_win32->hwnd, stage_win32->wtitle);
|
|
|
|
}
|
|
|
|
|
2010-01-15 22:56:37 +00:00
|
|
|
void
|
|
|
|
_clutter_stage_win32_update_cursor (ClutterStageWin32 *stage_win32)
|
|
|
|
{
|
|
|
|
HCURSOR cursor;
|
|
|
|
|
|
|
|
if (stage_win32->is_cursor_visible)
|
2010-04-09 14:43:42 +00:00
|
|
|
cursor = (HCURSOR) GetClassLongPtrW (stage_win32->hwnd, GCLP_HCURSOR);
|
2010-01-15 22:56:37 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
ClutterBackend *backend = clutter_get_default_backend ();
|
|
|
|
/* The documentation implies that we can just use
|
|
|
|
SetCursor(NULL) to get rid of the cursor but apparently this
|
|
|
|
doesn't work very well so instead we create an invisible
|
|
|
|
cursor */
|
|
|
|
cursor = _clutter_backend_win32_get_invisible_cursor (backend);
|
|
|
|
}
|
|
|
|
|
|
|
|
SetCursor (cursor);
|
|
|
|
}
|
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
static void
|
|
|
|
clutter_stage_win32_set_cursor_visible (ClutterStageWindow *stage_window,
|
|
|
|
gboolean cursor_visible)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
|
2010-01-15 22:56:37 +00:00
|
|
|
if (stage_win32->is_cursor_visible != cursor_visible)
|
|
|
|
{
|
|
|
|
POINT cursor_pos;
|
|
|
|
RECT client_rect;
|
|
|
|
|
|
|
|
stage_win32->is_cursor_visible = cursor_visible;
|
|
|
|
|
|
|
|
/* If the cursor is already over the client area of the window
|
|
|
|
then we need to update it immediately */
|
|
|
|
GetCursorPos (&cursor_pos);
|
|
|
|
if (WindowFromPoint (cursor_pos) == stage_win32->hwnd &&
|
|
|
|
ScreenToClient (stage_win32->hwnd, &cursor_pos) &&
|
|
|
|
GetClientRect (stage_win32->hwnd, &client_rect) &&
|
|
|
|
cursor_pos.x >= client_rect.left &&
|
|
|
|
cursor_pos.y >= client_rect.top &&
|
|
|
|
cursor_pos.x < client_rect.right &&
|
|
|
|
cursor_pos.y < client_rect.bottom)
|
|
|
|
_clutter_stage_win32_update_cursor (stage_win32);
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static LONG
|
|
|
|
get_window_style (ClutterStageWin32 *stage_win32)
|
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
ClutterStage *wrapper = stage_win32->wrapper;
|
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
/* Fullscreen mode shouldn't have any borders */
|
|
|
|
if ((stage_win32->state & CLUTTER_STAGE_STATE_FULLSCREEN))
|
|
|
|
return WS_POPUP;
|
|
|
|
/* Otherwise it's an overlapped window but if it isn't resizable
|
|
|
|
then it shouldn't have a thick frame */
|
2008-04-13 08:43:32 +00:00
|
|
|
else if (clutter_stage_get_user_resizable (wrapper))
|
2008-03-25 15:42:50 +00:00
|
|
|
return WS_OVERLAPPEDWINDOW;
|
|
|
|
else
|
2008-09-16 16:00:31 +00:00
|
|
|
return WS_OVERLAPPEDWINDOW & ~WS_THICKFRAME & ~WS_MAXIMIZEBOX;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2008-04-13 08:43:32 +00:00
|
|
|
clutter_stage_win32_set_user_resize (ClutterStageWindow *stage_window,
|
|
|
|
gboolean value)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
HWND hwnd = CLUTTER_STAGE_WIN32 (stage_window)->hwnd;
|
2008-03-25 15:42:50 +00:00
|
|
|
LONG old_style = GetWindowLongW (hwnd, GWL_STYLE);
|
|
|
|
|
|
|
|
/* Update the window style but preserve the visibility */
|
|
|
|
SetWindowLongW (hwnd, GWL_STYLE,
|
2008-04-13 08:43:32 +00:00
|
|
|
get_window_style (CLUTTER_STAGE_WIN32 (stage_window))
|
2008-03-25 15:42:50 +00:00
|
|
|
| (old_style & WS_VISIBLE));
|
2008-09-16 16:00:31 +00:00
|
|
|
/* Queue a redraw of the frame */
|
|
|
|
RedrawWindow (hwnd, NULL, NULL, RDW_FRAME | RDW_INVALIDATE);
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
2010-12-28 18:04:00 +00:00
|
|
|
static void
|
|
|
|
clutter_stage_win32_set_accept_focus (ClutterStageWindow *stage_window,
|
|
|
|
gboolean accept_focus)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
|
|
|
|
accept_focus = !!accept_focus;
|
|
|
|
|
|
|
|
stage_win32->accept_focus = accept_focus;
|
|
|
|
}
|
|
|
|
|
2008-04-13 08:43:32 +00:00
|
|
|
static ClutterActor *
|
|
|
|
clutter_stage_win32_get_wrapper (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
return CLUTTER_ACTOR (CLUTTER_STAGE_WIN32 (stage_window)->wrapper);
|
|
|
|
}
|
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
static void
|
2008-04-13 08:43:32 +00:00
|
|
|
clutter_stage_win32_set_fullscreen (ClutterStageWindow *stage_window,
|
|
|
|
gboolean value)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
HWND hwnd = CLUTTER_STAGE_WIN32 (stage_window)->hwnd;
|
2008-03-25 15:42:50 +00:00
|
|
|
LONG old_style = GetWindowLongW (hwnd, GWL_STYLE);
|
|
|
|
ClutterStageStateEvent event;
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
stage_win32->state |= CLUTTER_STAGE_STATE_FULLSCREEN;
|
|
|
|
else
|
|
|
|
stage_win32->state &= ~CLUTTER_STAGE_STATE_FULLSCREEN;
|
|
|
|
|
|
|
|
if (hwnd)
|
|
|
|
{
|
|
|
|
/* Update the window style but preserve the visibility */
|
|
|
|
SetWindowLongW (hwnd, GWL_STYLE,
|
|
|
|
get_window_style (stage_win32)
|
|
|
|
| (old_style & WS_VISIBLE));
|
|
|
|
/* Update the window size */
|
|
|
|
if (value)
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
get_fullscreen_rect (stage_win32);
|
|
|
|
SetWindowPos (hwnd, HWND_TOP,
|
|
|
|
stage_win32->fullscreen_rect.left,
|
|
|
|
stage_win32->fullscreen_rect.top,
|
|
|
|
stage_win32->fullscreen_rect.right
|
|
|
|
- stage_win32->fullscreen_rect.left,
|
|
|
|
stage_win32->fullscreen_rect.bottom
|
|
|
|
- stage_win32->fullscreen_rect.top,
|
|
|
|
0);
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
else
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
int full_width, full_height;
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
get_full_window_size (stage_win32,
|
|
|
|
stage_win32->win_width,
|
|
|
|
stage_win32->win_height,
|
|
|
|
&full_width, &full_height);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
SetWindowPos (stage_win32->hwnd, NULL,
|
|
|
|
0, 0,
|
|
|
|
full_width, full_height,
|
|
|
|
SWP_NOZORDER | SWP_NOMOVE);
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Report the state change */
|
|
|
|
memset (&event, 0, sizeof (event));
|
|
|
|
event.type = CLUTTER_STAGE_STATE;
|
2008-04-13 08:43:32 +00:00
|
|
|
event.stage = CLUTTER_STAGE (stage_win32->wrapper);
|
2008-03-25 15:42:50 +00:00
|
|
|
event.new_state = stage_win32->state;
|
|
|
|
event.changed_mask = CLUTTER_STAGE_STATE_FULLSCREEN;
|
|
|
|
clutter_event_put ((ClutterEvent *) &event);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ATOM
|
|
|
|
clutter_stage_win32_get_window_class ()
|
|
|
|
{
|
|
|
|
static ATOM klass = 0;
|
|
|
|
|
|
|
|
if (klass == 0)
|
|
|
|
{
|
|
|
|
WNDCLASSW wndclass;
|
|
|
|
memset (&wndclass, 0, sizeof (wndclass));
|
|
|
|
wndclass.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW;
|
|
|
|
wndclass.lpfnWndProc = _clutter_stage_win32_window_proc;
|
|
|
|
wndclass.cbWndExtra = sizeof (LONG_PTR);
|
|
|
|
wndclass.hInstance = GetModuleHandleW (NULL);
|
|
|
|
wndclass.hIcon = LoadIconW (NULL, (LPWSTR) IDI_APPLICATION);
|
|
|
|
wndclass.hCursor = LoadCursorW (NULL, (LPWSTR) IDC_ARROW);
|
|
|
|
wndclass.hbrBackground = NULL;
|
|
|
|
wndclass.lpszMenuName = NULL;
|
|
|
|
wndclass.lpszClassName = L"ClutterStageWin32";
|
|
|
|
klass = RegisterClassW (&wndclass);
|
|
|
|
}
|
|
|
|
|
|
|
|
return klass;
|
|
|
|
}
|
2008-06-25 12:25:43 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
static gboolean
|
|
|
|
clutter_stage_win32_realize (ClutterStageWindow *stage_window)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2009-10-26 15:28:36 +00:00
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
2011-04-21 15:52:54 +00:00
|
|
|
ClutterBackend *backend;
|
2008-03-30 16:51:01 +00:00
|
|
|
ClutterBackendWin32 *backend_win32;
|
2011-04-21 15:52:54 +00:00
|
|
|
CoglFramebuffer *framebuffer;
|
|
|
|
gfloat width;
|
|
|
|
gfloat height;
|
2010-01-20 16:41:25 +00:00
|
|
|
GError *error = NULL;
|
2011-04-21 15:52:54 +00:00
|
|
|
const char *clutter_vblank;
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
CLUTTER_NOTE (MISC, "Realizing main stage");
|
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
backend = CLUTTER_BACKEND (stage_win32->backend);
|
|
|
|
backend_win32 = CLUTTER_BACKEND_WIN32 (backend);
|
|
|
|
|
|
|
|
clutter_actor_get_size (CLUTTER_ACTOR (stage_win32->wrapper),
|
|
|
|
&width, &height);
|
|
|
|
|
|
|
|
stage_win32->onscreen = cogl_onscreen_new (backend->cogl_context,
|
|
|
|
width, height);
|
2008-03-30 16:51:01 +00:00
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
if (stage_win32->hwnd == NULL)
|
|
|
|
{
|
|
|
|
ATOM window_class = clutter_stage_win32_get_window_class ();
|
|
|
|
int win_xpos, win_ypos, win_width, win_height;
|
|
|
|
|
|
|
|
if (window_class == 0)
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
2008-03-25 15:42:50 +00:00
|
|
|
g_critical ("Unable to register window class");
|
2009-10-26 15:28:36 +00:00
|
|
|
goto fail;
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
/* If we're in fullscreen mode then use the fullscreen rect
|
2009-10-26 15:28:36 +00:00
|
|
|
instead */
|
2008-03-25 15:42:50 +00:00
|
|
|
if ((stage_win32->state & CLUTTER_STAGE_STATE_FULLSCREEN))
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
get_fullscreen_rect (stage_win32);
|
|
|
|
win_xpos = stage_win32->fullscreen_rect.left;
|
|
|
|
win_ypos = stage_win32->fullscreen_rect.top;
|
|
|
|
win_width = stage_win32->fullscreen_rect.right - win_xpos;
|
2009-11-27 15:53:50 +00:00
|
|
|
win_height = stage_win32->fullscreen_rect.bottom - win_ypos;
|
2009-10-26 15:28:36 +00:00
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
else
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
win_xpos = win_ypos = CW_USEDEFAULT;
|
2008-06-12 12:12:25 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
get_full_window_size (stage_win32,
|
|
|
|
stage_win32->win_width,
|
|
|
|
stage_win32->win_height,
|
|
|
|
&win_width, &win_height);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stage_win32->wtitle == NULL)
|
|
|
|
stage_win32->wtitle = g_utf8_to_utf16 (".", -1, NULL, NULL, NULL);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
stage_win32->hwnd = CreateWindowW ((LPWSTR) MAKEINTATOM (window_class),
|
2009-10-26 15:28:36 +00:00
|
|
|
stage_win32->wtitle,
|
2008-03-25 15:42:50 +00:00
|
|
|
get_window_style (stage_win32),
|
|
|
|
win_xpos,
|
|
|
|
win_ypos,
|
|
|
|
win_width,
|
|
|
|
win_height,
|
|
|
|
NULL, NULL,
|
|
|
|
GetModuleHandle (NULL),
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (stage_win32->hwnd == NULL)
|
2009-10-26 15:28:36 +00:00
|
|
|
{
|
|
|
|
g_critical ("Unable to create stage window");
|
|
|
|
goto fail;
|
|
|
|
}
|
2008-03-25 15:42:50 +00:00
|
|
|
|
|
|
|
/* Store a pointer to the actor in the extra bytes of the window
|
2009-10-26 15:28:36 +00:00
|
|
|
so we can quickly access it in the window procedure */
|
2008-03-25 15:42:50 +00:00
|
|
|
SetWindowLongPtrW (stage_win32->hwnd, 0, (LONG_PTR) stage_win32);
|
|
|
|
}
|
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
cogl_onscreen_win32_set_foreign_window (stage_win32->onscreen,
|
|
|
|
stage_win32->hwnd);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
clutter_vblank = _clutter_backend_win32_get_vblank ();
|
|
|
|
if (clutter_vblank && strcmp (clutter_vblank, "none") == 0)
|
|
|
|
cogl_onscreen_set_swap_throttled (stage_win32->onscreen, FALSE);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
framebuffer = COGL_FRAMEBUFFER (stage_win32->onscreen);
|
|
|
|
if (!cogl_framebuffer_allocate (framebuffer, &error))
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2011-04-21 15:52:54 +00:00
|
|
|
g_warning ("Failed to allocate stage: %s", error->message);
|
2010-01-20 16:41:25 +00:00
|
|
|
g_error_free (error);
|
2011-04-21 15:52:54 +00:00
|
|
|
cogl_object_unref (stage_win32->onscreen);
|
|
|
|
stage_win32->onscreen = NULL;
|
2008-05-15 22:03:22 +00:00
|
|
|
goto fail;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
/* Create a context. This will be a no-op if we already have one */
|
|
|
|
if (!_clutter_backend_create_context (CLUTTER_BACKEND (backend_win32),
|
|
|
|
&error))
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2011-04-21 15:52:54 +00:00
|
|
|
g_critical ("Unable to realize stage: %s", error->message);
|
|
|
|
g_error_free (error);
|
2010-01-20 16:41:25 +00:00
|
|
|
goto fail;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
2009-06-06 13:55:00 +00:00
|
|
|
CLUTTER_NOTE (BACKEND, "Successfully realized stage");
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
return TRUE;
|
2008-05-15 22:03:22 +00:00
|
|
|
|
|
|
|
fail:
|
2009-10-26 15:28:36 +00:00
|
|
|
return FALSE;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-10-26 15:28:36 +00:00
|
|
|
clutter_stage_win32_unprepare_window (ClutterStageWin32 *stage_win32)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-04-15 16:12:37 +00:00
|
|
|
if (!stage_win32->is_foreign_win && stage_win32->hwnd)
|
2008-03-25 15:42:50 +00:00
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
/* Drop the pointer to this stage in the window so that any
|
2009-10-26 15:28:36 +00:00
|
|
|
further messages won't be processed. The stage might be being
|
|
|
|
destroyed so otherwise the messages would be handled with an
|
|
|
|
invalid stage instance */
|
2008-04-13 08:43:32 +00:00
|
|
|
SetWindowLongPtrW (stage_win32->hwnd, 0, (LONG_PTR) 0);
|
2008-03-25 15:42:50 +00:00
|
|
|
DestroyWindow (stage_win32->hwnd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
static void
|
|
|
|
clutter_stage_win32_unrealize (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
|
|
|
|
CLUTTER_NOTE (BACKEND, "Unrealizing stage");
|
|
|
|
|
|
|
|
clutter_stage_win32_unprepare_window (stage_win32);
|
|
|
|
}
|
|
|
|
|
2011-02-04 15:09:41 +00:00
|
|
|
static void
|
|
|
|
clutter_stage_win32_redraw (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (stage_window);
|
|
|
|
|
|
|
|
/* this will cause the stage implementation to be painted */
|
|
|
|
_clutter_stage_do_paint (stage_win32->wrapper, NULL);
|
|
|
|
cogl_flush ();
|
|
|
|
|
2011-04-21 15:52:54 +00:00
|
|
|
if (stage_win32->onscreen)
|
|
|
|
cogl_framebuffer_swap_buffers (COGL_FRAMEBUFFER (stage_win32->onscreen));
|
2011-02-04 15:09:41 +00:00
|
|
|
}
|
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
static void
|
|
|
|
clutter_stage_win32_dispose (GObject *gobject)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32 = CLUTTER_STAGE_WIN32 (gobject);
|
|
|
|
|
2009-10-26 15:28:36 +00:00
|
|
|
/* Make sure that context and window are destroyed in case unrealize
|
|
|
|
* hasn't been called yet.
|
|
|
|
*/
|
|
|
|
if (stage_win32->hwnd)
|
|
|
|
clutter_stage_win32_unprepare_window (stage_win32);
|
|
|
|
|
|
|
|
if (stage_win32->wtitle)
|
2011-04-21 15:52:54 +00:00
|
|
|
{
|
|
|
|
g_free (stage_win32->wtitle);
|
|
|
|
stage_win32->wtitle = NULL;
|
|
|
|
}
|
2009-10-26 15:28:36 +00:00
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
G_OBJECT_CLASS (clutter_stage_win32_parent_class)->dispose (gobject);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clutter_stage_win32_class_init (ClutterStageWin32Class *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
gobject_class->dispose = clutter_stage_win32_dispose;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clutter_stage_win32_init (ClutterStageWin32 *stage)
|
|
|
|
{
|
|
|
|
stage->hwnd = NULL;
|
|
|
|
stage->win_width = 640;
|
|
|
|
stage->win_height = 480;
|
|
|
|
stage->backend = NULL;
|
|
|
|
stage->scroll_pos = 0;
|
2009-10-26 15:28:36 +00:00
|
|
|
stage->wtitle = NULL;
|
2008-04-13 08:43:32 +00:00
|
|
|
stage->wrapper = NULL;
|
2010-12-28 18:04:00 +00:00
|
|
|
|
|
|
|
stage->is_foreign_win = FALSE;
|
|
|
|
stage->is_cursor_visible = TRUE;
|
|
|
|
stage->accept_focus = TRUE;
|
2008-04-13 08:43:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clutter_stage_window_iface_init (ClutterStageWindowIface *iface)
|
|
|
|
{
|
|
|
|
iface->get_wrapper = clutter_stage_win32_get_wrapper;
|
|
|
|
iface->set_title = clutter_stage_win32_set_title;
|
|
|
|
iface->set_fullscreen = clutter_stage_win32_set_fullscreen;
|
2009-10-26 15:28:36 +00:00
|
|
|
iface->set_cursor_visible = clutter_stage_win32_set_cursor_visible;
|
2008-04-13 08:43:32 +00:00
|
|
|
iface->set_user_resizable = clutter_stage_win32_set_user_resize;
|
2010-12-28 18:04:00 +00:00
|
|
|
iface->set_accept_focus = clutter_stage_win32_set_accept_focus;
|
2009-07-28 11:00:49 +00:00
|
|
|
iface->show = clutter_stage_win32_show;
|
|
|
|
iface->hide = clutter_stage_win32_hide;
|
2009-10-26 15:28:36 +00:00
|
|
|
iface->resize = clutter_stage_win32_resize;
|
|
|
|
iface->get_geometry = clutter_stage_win32_get_geometry;
|
|
|
|
iface->realize = clutter_stage_win32_realize;
|
|
|
|
iface->unrealize = clutter_stage_win32_unrealize;
|
2011-02-04 15:09:41 +00:00
|
|
|
iface->redraw = clutter_stage_win32_redraw;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* clutter_win32_get_stage_window:
|
|
|
|
* @stage: a #ClutterStage
|
|
|
|
*
|
2008-03-26 22:22:32 +00:00
|
|
|
* Gets the stage's window handle
|
2008-03-25 15:42:50 +00:00
|
|
|
*
|
|
|
|
* Return value: An HWND for the stage window.
|
|
|
|
*
|
|
|
|
* Since: 0.8
|
|
|
|
*/
|
|
|
|
HWND
|
|
|
|
clutter_win32_get_stage_window (ClutterStage *stage)
|
|
|
|
{
|
2008-04-13 08:43:32 +00:00
|
|
|
ClutterStageWindow *impl;
|
|
|
|
|
|
|
|
g_return_val_if_fail (CLUTTER_IS_STAGE (stage), NULL);
|
|
|
|
|
|
|
|
impl = _clutter_stage_get_window (stage);
|
|
|
|
|
|
|
|
g_return_val_if_fail (CLUTTER_IS_STAGE_WIN32 (impl), NULL);
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2008-04-13 08:43:32 +00:00
|
|
|
return CLUTTER_STAGE_WIN32 (impl)->hwnd;
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
2008-03-30 16:51:01 +00:00
|
|
|
/**
|
|
|
|
* clutter_win32_get_stage_from_window:
|
|
|
|
* @hwnd: a window handle
|
|
|
|
*
|
|
|
|
* Gets the stage for a particular window.
|
|
|
|
*
|
|
|
|
* Return value: The stage or NULL if a stage does not exist for the
|
|
|
|
* window.
|
|
|
|
*
|
|
|
|
* Since: 0.8
|
|
|
|
*/
|
|
|
|
ClutterStage *
|
|
|
|
clutter_win32_get_stage_from_window (HWND hwnd)
|
|
|
|
{
|
|
|
|
/* Check whether the window handle is an instance of the stage
|
|
|
|
window class */
|
|
|
|
if ((ATOM) GetClassLongPtrW (hwnd, GCW_ATOM)
|
|
|
|
== clutter_stage_win32_get_window_class ())
|
|
|
|
/* If it is there should be a pointer to the stage in the window
|
|
|
|
extra data */
|
2008-04-13 08:43:32 +00:00
|
|
|
return CLUTTER_STAGE_WIN32 (GetWindowLongPtrW (hwnd, 0))->wrapper;
|
2008-03-30 16:51:01 +00:00
|
|
|
else
|
2008-04-15 16:12:37 +00:00
|
|
|
{
|
|
|
|
/* Otherwise it might be a foreign window so we should check the
|
|
|
|
stage list */
|
2009-07-13 16:04:05 +00:00
|
|
|
ClutterStageManager *stage_manager;
|
2009-06-17 16:59:54 +00:00
|
|
|
const GSList *stages, *l;
|
2008-04-15 16:12:37 +00:00
|
|
|
|
2009-06-17 16:59:54 +00:00
|
|
|
stage_manager = clutter_stage_manager_get_default ();
|
|
|
|
stages = clutter_stage_manager_peek_stages (stage_manager);
|
|
|
|
|
|
|
|
for (l = stages; l != NULL; l = l->next)
|
2008-04-15 16:12:37 +00:00
|
|
|
{
|
|
|
|
ClutterStage *stage = l->data;
|
|
|
|
ClutterStageWindow *impl;
|
|
|
|
|
|
|
|
impl = _clutter_stage_get_window (stage);
|
|
|
|
g_assert (CLUTTER_IS_STAGE_WIN32 (impl));
|
|
|
|
|
|
|
|
if (CLUTTER_STAGE_WIN32 (impl)->hwnd == hwnd)
|
|
|
|
return stage;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2008-03-30 16:51:01 +00:00
|
|
|
}
|
2009-10-26 15:28:36 +00:00
|
|
|
|
2010-10-14 13:30:54 +00:00
|
|
|
typedef struct {
|
|
|
|
ClutterStageWin32 *stage_win32;
|
|
|
|
ClutterGeometry geom;
|
|
|
|
HWND hwnd;
|
|
|
|
guint destroy_old_hwnd : 1;
|
|
|
|
} ForeignWindowData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_foreign_window_callback (ClutterActor *actor,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
ForeignWindowData *fwd = data;
|
|
|
|
|
|
|
|
CLUTTER_NOTE (BACKEND, "Setting foreign window (0x%x)",
|
|
|
|
(guint) fwd->hwnd);
|
|
|
|
|
|
|
|
if (fwd->destroy_old_hwnd && fwd->stage_win32->hwnd != NULL)
|
|
|
|
{
|
|
|
|
CLUTTER_NOTE (BACKEND, "Destroying previous window (0x%x)",
|
|
|
|
(guint) fwd->stage_win32->hwnd);
|
|
|
|
DestroyWindow (fwd->stage_win32->hwnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
fwd->stage_win32->hwnd = fwd->hwnd;
|
|
|
|
fwd->stage_win32->is_foreign_win = TRUE;
|
|
|
|
|
|
|
|
fwd->stage_win32->win_width = fwd->geom.width;
|
|
|
|
fwd->stage_win32->win_height = fwd->geom.height;
|
|
|
|
|
|
|
|
clutter_actor_set_geometry (actor, &fwd->geom);
|
|
|
|
|
|
|
|
/* calling this with the stage unrealized will unset the stage
|
|
|
|
* from the GL context; once the stage is realized the GL context
|
|
|
|
* will be set again
|
|
|
|
*/
|
|
|
|
clutter_stage_ensure_current (CLUTTER_STAGE (actor));
|
|
|
|
}
|
|
|
|
|
2008-04-15 16:12:37 +00:00
|
|
|
/**
|
|
|
|
* clutter_win32_set_stage_foreign:
|
|
|
|
* @stage: a #ClutterStage
|
|
|
|
* @hwnd: an existing window handle
|
|
|
|
*
|
|
|
|
* Target the #ClutterStage to use an existing external window handle.
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if foreign window is valid
|
|
|
|
*
|
|
|
|
* Since: 0.8
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
clutter_win32_set_stage_foreign (ClutterStage *stage,
|
|
|
|
HWND hwnd)
|
|
|
|
{
|
|
|
|
ClutterStageWin32 *stage_win32;
|
|
|
|
ClutterStageWindow *impl;
|
|
|
|
ClutterActor *actor;
|
|
|
|
RECT client_rect;
|
2010-10-14 13:30:54 +00:00
|
|
|
ForeignWindowData fwd;
|
2008-04-15 16:12:37 +00:00
|
|
|
|
|
|
|
g_return_val_if_fail (CLUTTER_IS_STAGE (stage), FALSE);
|
|
|
|
g_return_val_if_fail (hwnd != NULL, FALSE);
|
|
|
|
|
|
|
|
actor = CLUTTER_ACTOR (stage);
|
|
|
|
|
|
|
|
impl = _clutter_stage_get_window (stage);
|
|
|
|
stage_win32 = CLUTTER_STAGE_WIN32 (impl);
|
|
|
|
|
|
|
|
if (!GetClientRect (hwnd, &client_rect))
|
|
|
|
{
|
|
|
|
g_warning ("Unable to retrieve the new window geometry");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-10-14 13:30:54 +00:00
|
|
|
fwd.stage_win32 = stage_win32;
|
|
|
|
fwd.hwnd = hwnd;
|
2008-04-15 16:12:37 +00:00
|
|
|
|
2010-10-14 13:30:54 +00:00
|
|
|
/* destroy the old HWND, if we have one and it's ours */
|
|
|
|
if (stage_win32->hwnd != NULL && !stage_win32->is_foreign_win)
|
|
|
|
fwd.destroy_old_hwnd = TRUE;
|
|
|
|
else
|
|
|
|
fwd.destroy_old_hwnd = FALSE;
|
|
|
|
|
|
|
|
fwd.geom.x = 0;
|
|
|
|
fwd.geom.y = 0;
|
|
|
|
fwd.geom.width = client_rect.right - client_rect.left;
|
|
|
|
fwd.geom.height = client_rect.bottom - client_rect.top;
|
|
|
|
|
|
|
|
_clutter_actor_rerealize (actor,
|
|
|
|
set_foreign_window_callback,
|
|
|
|
&fwd);
|
|
|
|
|
|
|
|
/* Queue a relayout - so the stage will be allocated the new
|
|
|
|
* window size.
|
|
|
|
*
|
|
|
|
* Note also that when the stage gets allocated the new
|
|
|
|
* window size that will result in the stage's
|
|
|
|
* priv->viewport being changed, which will in turn result
|
|
|
|
* in the Cogl viewport changing when _clutter_do_redraw
|
|
|
|
* calls _clutter_stage_maybe_setup_viewport().
|
|
|
|
*/
|
|
|
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (stage));
|
2008-04-15 16:12:37 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2008-03-25 15:42:50 +00:00
|
|
|
void
|
|
|
|
clutter_stage_win32_map (ClutterStageWin32 *stage_win32)
|
|
|
|
{
|
2009-06-04 10:59:17 +00:00
|
|
|
clutter_actor_map (CLUTTER_ACTOR (stage_win32->wrapper));
|
2008-03-25 15:42:50 +00:00
|
|
|
|
2008-06-12 12:12:25 +00:00
|
|
|
clutter_actor_queue_relayout (CLUTTER_ACTOR (stage_win32->wrapper));
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
clutter_stage_win32_unmap (ClutterStageWin32 *stage_win32)
|
|
|
|
{
|
2009-06-04 10:59:17 +00:00
|
|
|
clutter_actor_unmap (CLUTTER_ACTOR (stage_win32->wrapper));
|
2008-03-25 15:42:50 +00:00
|
|
|
}
|