转载时请注明出处和作者联系方式
作者联系方式:李先静 <xianjimli at hotmail dot com>
虽然我们实现了单实例应用程序,即在应用程序第二次运行时自动激活第一个实例,并将参数传递给第一个运行实例。但如果在桌面连续点击应用程序的起动图标,会让应用程序起动过过程变得更慢,为了避免这种情况出现,有必要实现类似WaitCursor机制。GTK+-2.6/DFB中没有实现WaitCursor,新版本中好像也没有,只好自己动手了:
- 在gdkwindow.h文件中增加两个函数:
gint gdk_window_begin_wait_cursor(GdkWindow *window); gint gdk_window_end_wait_cursor(GdkWindow *window);
- 在_GdkWindowObject结构中增加waitcursor的信息。
struct
{
gint ref;
gint elapsed;
gint timeout_id;
}wait_cursor;- 在gdkwindow-directfb.c中实现这两个函数,很简单,就是实现一个定时器,每隔一段时间更新一下Cursor。
gint gdk_window_begin_wait_cursor(GdkWindow *window)
{
GdkWindowObject *private = GDK_WINDOW_OBJECT (window);
g_return_val_if_fail(private != NULL, 0);
private->wait_cursor.ref++;
if(private->wait_cursor.ref == 1 && update_wait_cursor(window))
{
private->wait_cursor.elapsed = 0;
private->wait_cursor.timeout_id = g_timeout_add(250, update_wait_cursor, window);
#ifndef LINUX_I386
gdk_directfb_display->layer->EnableCursor(gdk_directfb_display->layer, 1);
#endif
}
return private->wait_cursor.ref;
}
gint gdk_window_end_wait_cursor(GdkWindow *window)
{
GdkWindowObject *private = GDK_WINDOW_OBJECT (window);
if(private == NULL || private->wait_cursor.ref <=0)
{
return 0;
}
private->wait_cursor.ref--;
#ifndef LINUX_I386
gdk_directfb_display->layer->EnableCursor(gdk_directfb_display->layer, private->wait_cursor.ref != 0);
#endif
return private->wait_cursor.ref;
}
static gboolean update_wait_cursor(gpointer user_data)
{
GdkWindow *window = user_data;
GdkWindowObject *private = GDK_WINDOW_OBJECT (window);
g_return_val_if_fail(private != NULL, FALSE);
int cursor_type = GDK_WAIT_BEGIN +
(private->wait_cursor.elapsed)%(GDK_WAIT_END - GDK_WAIT_BEGIN + 1);
GdkCursor* cursor = gdk_cursor_new_for_display (gdk_display_get_default(),
cursor_type);
gdk_window_set_cursor(window, cursor);
gdk_cursor_unref(cursor);
private->wait_cursor.elapsed++;
if(private->wait_cursor.ref <= 0)
{
private->wait_cursor.elapsed = 0;
private->wait_cursor.timeout_id = 0;
gdk_window_set_cursor(window, NULL);
}
return private->wait_cursor.ref > 0;
}在gdkcursor-directfb.c中加载Cursor图片,Cursor图片与主题关联:
static GdkCursor *
gdk_cursor_load_wait_cursor(GdkDisplay *display,
GdkCursorType cursor_type)
{
GdkCursor* cursor = NULL;
if(cursor_type >= GDK_WAIT_BEGIN && cursor_type <= GDK_WAIT_END)
{
int index = 0;
GValue value = {0};
char* theme = NULL;
char filename[260] = {0};
GObject* setting = g_object_get_data (G_OBJECT (gdk_screen_get_default()),
"gtk-settings");
g_return_val_if_fail(setting != NULL, NULL);
g_value_init(&value, G_TYPE_STRING);
g_object_get_property (setting, "gtk-icon-theme-name", &value);
theme = g_value_get_string(&value);
index = cursor_type-GDK_WAIT_BEGIN + 1;
snprintf(filename, sizeof(filename), WAITCURSOR, theme, index);
if((cursor = gdk_cursor_new_from_name(display, filename)) == NULL)
{
snprintf(filename, sizeof(filename), WAITCURSOR, "hicolor", index);
cursor = gdk_cursor_new_from_name(display, filename);
g_debug("load %p %s", cursor, filename);
}
g_value_reset(&value);
}
return cursor;
}
- 要注意的是gdk_cursor_new_from_pixbuf有点问题,pixbuf中的格式居然是GBRA而不是代码中要求的ARGB,要修改一下这个函数,否则无法实现透明效果。
- 为了避免enable/disable时闪现一下传统的cursor,修改gdk_cursor_new_for_display,让alpha=0,让传统cursor全透明。
- 出现WaitCursor时,丢弃按键和笔点事件,修改gdk_event_translate:
if(private->wait_cursor.ref > 0)
{
switch (dfbevent->type)
{
case DWET_BUTTONDOWN:
case DWET_BUTTONUP:
case DWET_MOTION:
case DWET_KEYDOWN:
case DWET_KEYUP:
return NULL;
default:break;
}
}在手机上和PC上有点不问,手机上平时不能显示cursor,而且waitcursor的位置要居中,所以还要修改DirectFB:
- 在dfb_layer_context_create_window中禁止cursor,把以前的dfb_windowstack_cursor_enable(core, stack, true )改为dfb_windowstack_cursor_enable( core, stack, false )
- 在wm_update_cursor里让位置居中:
#ifndef LINUX_I386
if(context->stack->cursor.enabled && (flags & CCUF_POSITION))
{
stack->cursor.hot.x = stack->cursor.x - (context->width/2 -
stack->cursor.size.w);
stack->cursor.hot.y = stack->cursor.y - (context->height/2 -
stack->cursor.size.h);
}
#endif测试了一下,基本功能正常。
~~end~~