[PATCH] gvncviewer: add options for disabling relative mouse extension and keyboard grab
- From: Michal Suchanek <hramrach centrum cz>
- To: gtk-vnc-list gnome org
- Subject: [PATCH] gvncviewer: add options for disabling relative mouse extension and keyboard grab
- Date: Wed, 9 Sep 2009 00:18:05 +0200
Hello
This is a single patch that adds both options allowing quickly
switching to and from the viewer.
Also attached at http://bugzilla.gnome.org/show_bug.cgi?id=594566
Thanks
Michal
diff --git a/examples/gvncviewer.c b/examples/gvncviewer.c
index c3fbd74..960b4cc 100644
--- a/examples/gvncviewer.c
+++ b/examples/gvncviewer.c
@@ -32,8 +32,18 @@
#endif
static gchar **args = NULL;
+static gboolean keyboard_grab = TRUE;
+static gboolean relative_pointer = TRUE;
static const GOptionEntry options [] =
{
+ { "absolute", 'a', G_OPTION_FLAG_REVERSE, G_OPTION_ARG_NONE,
+ &relative_pointer, "Turn off relative pointer extension", NULL },
+ { "relative", 'r', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE,
+ &relative_pointer, "Turn on relative pointer extension", NULL },
+ { "no-keyboard-grab", 'k', G_OPTION_FLAG_REVERSE, G_OPTION_ARG_NONE,
+ &keyboard_grab, "Turn off keyboard grab", NULL },
+ { "keyboard-grab", '\0', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE,
+ &keyboard_grab, NULL, NULL },
{
G_OPTION_REMAINING, '\0', 0, G_OPTION_ARG_STRING_ARRAY, &args,
NULL, "[hostname][:display]" },
@@ -216,6 +226,26 @@ static void do_scaling(GtkWidget *menu, GtkWidget *vncdisplay)
vnc_display_set_scaling(VNC_DISPLAY(vncdisplay), FALSE);
}
+static void do_keyboard_grab(GtkWidget *menu, GtkWidget *vncdisplay)
+{
+ if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menu)))
+ keyboard_grab = TRUE;
+ else
+ keyboard_grab = FALSE;
+
+ vnc_display_set_keyboard_grab(VNC_DISPLAY(vncdisplay), keyboard_grab);;
+}
+
+static void do_relative(GtkWidget *menu, GtkWidget *vncdisplay)
+{
+ if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menu)))
+ relative_pointer = TRUE;
+ else
+ relative_pointer = FALSE;
+
+ vnc_display_set_pointer_allow_relative(VNC_DISPLAY(vncdisplay), relative_pointer);
+}
+
static void vnc_credential(GtkWidget *vncdisplay, GValueArray *credList)
{
GtkWidget *dialog = NULL;
@@ -367,6 +397,8 @@ int main(int argc, char **argv)
GtkWidget *cab;
GtkWidget *fullscreen;
GtkWidget *scaling;
+ GtkWidget *grab_toggle;
+ GtkWidget *relative_toggle;
const char *help_msg = "Run 'gvncviewer --help' to see a full list of available command line options";
/* Setup command line options */
@@ -441,6 +473,14 @@ int main(int argc, char **argv)
gtk_menu_item_set_submenu(GTK_MENU_ITEM(view), submenu);
+ grab_toggle = gtk_check_menu_item_new_with_mnemonic("Keyboard _Grab");
+ gtk_menu_bar_append(GTK_MENU_BAR(menubar), grab_toggle);
+ gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(grab_toggle), keyboard_grab);
+
+ relative_toggle = gtk_check_menu_item_new_with_mnemonic("_Relative Pointer");
+ gtk_menu_bar_append(GTK_MENU_BAR(menubar), relative_toggle);
+ gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(relative_toggle), relative_pointer);
+
#if WITH_LIBVIEW
ViewAutoDrawer_SetActive(VIEW_AUTODRAWER(layout), FALSE);
ViewOvBox_SetOver(VIEW_OV_BOX(layout), menubar);
@@ -461,11 +501,12 @@ int main(int argc, char **argv)
} else
snprintf(port, sizeof(port), "%d", 5900);
- if (!*hostname)
+ if (!*hostname)
snprintf(hostname, sizeof(hostname), "%s", "127.0.0.1");
vnc_display_open_host(VNC_DISPLAY(vnc), hostname, port);
- vnc_display_set_keyboard_grab(VNC_DISPLAY(vnc), TRUE);
+ vnc_display_set_keyboard_grab(VNC_DISPLAY(vnc), keyboard_grab);;
vnc_display_set_pointer_grab(VNC_DISPLAY(vnc), TRUE);
+ vnc_display_set_pointer_allow_relative(VNC_DISPLAY(vnc), relative_pointer);
if (!gtk_widget_is_composited(window)) {
vnc_display_set_scaling(VNC_DISPLAY(vnc), TRUE);
@@ -520,6 +561,10 @@ int main(int argc, char **argv)
G_CALLBACK(do_fullscreen), window);
g_signal_connect(scaling, "toggled",
G_CALLBACK(do_scaling), vnc);
+ g_signal_connect(grab_toggle, "toggled",
+ G_CALLBACK(do_keyboard_grab), vnc);
+ g_signal_connect(relative_toggle, "toggled",
+ G_CALLBACK(do_relative), vnc);
#if WITH_LIBVIEW
g_signal_connect(window, "window-state-event",
G_CALLBACK(window_state_event), layout);
diff --git a/src/vncdisplay.c b/src/vncdisplay.c
index 1f080fe..757616c 100644
--- a/src/vncdisplay.c
+++ b/src/vncdisplay.c
@@ -80,6 +80,7 @@ struct _VncDisplayPrivate
int last_y;
gboolean absolute;
+ gboolean allow_relative;
gboolean grab_pointer;
gboolean grab_keyboard;
@@ -93,6 +94,11 @@ struct _VncDisplayPrivate
GSList *preferable_auths;
};
+inline static gboolean absolute(const struct _VncDisplayPrivate * priv)
+{
+ return priv->absolute || !priv->allow_relative;
+}
+
/* Delayed signal emission.
*
* We want signals to be delivered in the system coroutine. This helps avoid
@@ -131,7 +137,8 @@ enum
PROP_LOSSY_ENCODING,
PROP_SCALING,
PROP_SHARED_FLAG,
- PROP_FORCE_SIZE
+ PROP_FORCE_SIZE,
+ PROP_POINTER_ALLOW_RELATIVE
};
/* Signals */
@@ -190,6 +197,9 @@ vnc_display_get_property (GObject *object,
case PROP_KEYBOARD_GRAB:
g_value_set_boolean (value, vnc->priv->grab_keyboard);
break;
+ case PROP_POINTER_ALLOW_RELATIVE:
+ g_value_set_boolean (value, vnc->priv->allow_relative);
+ break;
case PROP_READ_ONLY:
g_value_set_boolean (value, vnc->priv->read_only);
break;
@@ -239,6 +249,8 @@ vnc_display_set_property (GObject *object,
case PROP_KEYBOARD_GRAB:
vnc_display_set_keyboard_grab (vnc, g_value_get_boolean (value));
break;
+ case PROP_POINTER_ALLOW_RELATIVE:
+ vnc_display_set_pointer_allow_relative (vnc, g_value_get_boolean (value));
case PROP_READ_ONLY:
vnc_display_set_read_only (vnc, g_value_get_boolean (value));
break;
@@ -475,7 +487,7 @@ static void do_pointer_ungrab(VncDisplay *obj, gboolean quiet)
gdk_pointer_ungrab(GDK_CURRENT_TIME);
priv->in_pointer_grab = FALSE;
- if (priv->absolute)
+ if (absolute(priv))
do_pointer_hide(obj);
if (!quiet)
@@ -503,7 +515,7 @@ static gboolean button_event(GtkWidget *widget, GdkEventButton *button)
gtk_widget_grab_focus (widget);
- if (priv->grab_pointer && !priv->absolute && !priv->in_pointer_grab &&
+ if (priv->grab_pointer && !absolute(priv) && !priv->in_pointer_grab &&
button->button == 1 && button->type == GDK_BUTTON_PRESS)
do_pointer_grab(VNC_DISPLAY(widget), FALSE);
@@ -513,7 +525,7 @@ static gboolean button_event(GtkWidget *widget, GdkEventButton *button)
else if (button->type == GDK_BUTTON_RELEASE)
priv->button_mask &= ~n;
- if (priv->absolute) {
+ if (absolute(priv)) {
gvnc_pointer_event(priv->gvnc, priv->button_mask,
priv->last_x, priv->last_y);
} else {
@@ -546,7 +558,7 @@ static gboolean scroll_event(GtkWidget *widget, GdkEventScroll *scroll)
else
return FALSE;
- if (priv->absolute) {
+ if (absolute(priv)) {
gvnc_pointer_event(priv->gvnc, priv->button_mask | mask,
priv->last_x, priv->last_y);
gvnc_pointer_event(priv->gvnc, priv->button_mask,
@@ -588,7 +600,7 @@ static gboolean motion_event(GtkWidget *widget, GdkEventMotion *motion)
/* In relative mode, only move the server mouse pointer
* if the client grab is active */
- if (!priv->absolute && !priv->in_pointer_grab)
+ if (!absolute(priv) && !priv->in_pointer_grab)
return FALSE;
if (priv->read_only)
@@ -622,7 +634,7 @@ static gboolean motion_event(GtkWidget *widget, GdkEventMotion *motion)
}
/* Next adjust the real client pointer */
- if (!priv->absolute) {
+ if (!absolute(priv)) {
GdkDrawable *drawable = GDK_DRAWABLE(widget->window);
GdkDisplay *display = gdk_drawable_get_display(drawable);
GdkScreen *screen = gdk_drawable_get_screen(drawable);
@@ -652,7 +664,7 @@ static gboolean motion_event(GtkWidget *widget, GdkEventMotion *motion)
/* Finally send the event to server */
if (priv->last_x != -1) {
int dx, dy;
- if (priv->absolute) {
+ if (absolute(priv)) {
dx = (int)motion->x;
dy = (int)motion->y;
@@ -756,7 +768,7 @@ static gboolean key_event(GtkWidget *widget, GdkEventKey *key)
(keyval == GDK_Alt_L && (key->state & GDK_CONTROL_MASK)))) {
if (priv->in_pointer_grab)
do_pointer_ungrab(VNC_DISPLAY(widget), FALSE);
- else if (!priv->grab_keyboard || !priv->absolute)
+ else if (!priv->grab_keyboard || !absolute(priv))
do_pointer_grab(VNC_DISPLAY(widget), FALSE);
}
@@ -980,7 +992,7 @@ static gboolean do_resize(void *opaque, int width, int height, gboolean quiet)
priv->null_cursor = create_null_cursor();
if (priv->local_pointer)
do_pointer_show(obj);
- else if (priv->in_pointer_grab || priv->absolute)
+ else if (priv->in_pointer_grab || absolute(priv))
do_pointer_hide(obj);
priv->gc = gdk_gc_new(GTK_WIDGET(obj)->window);
}
@@ -1034,17 +1046,17 @@ static gboolean on_get_preferred_pixel_format(void *opaque,
return TRUE;
}
-static gboolean on_pointer_type_change(void *opaque, int absolute)
+static gboolean on_pointer_type_change(void *opaque, int _absolute)
{
VncDisplay *obj = VNC_DISPLAY(opaque);
VncDisplayPrivate *priv = obj->priv;
- if (absolute && priv->in_pointer_grab && priv->grab_pointer)
- do_pointer_ungrab(obj, FALSE);
+ priv->absolute = _absolute;
- priv->absolute = absolute;
+ if (absolute(priv) && priv->in_pointer_grab && priv->grab_pointer)
+ do_pointer_ungrab(obj, FALSE);
- if (!priv->in_pointer_grab && !priv->absolute)
+ if (!priv->in_pointer_grab && !absolute(priv))
do_pointer_show(obj);
return TRUE;
@@ -1210,7 +1222,7 @@ static gboolean on_local_cursor(void *opaque, int x, int y, int width, int heigh
if (priv->in_pointer_grab) {
do_pointer_ungrab(obj, TRUE);
do_pointer_grab(obj, TRUE);
- } else if (priv->absolute) {
+ } else if (absolute(priv)) {
do_pointer_hide(obj);
}
@@ -1532,7 +1544,7 @@ void vnc_display_send_pointer(VncDisplay *obj, gint x, gint y, int button_mask)
if (priv->gvnc == NULL || !gvnc_is_open(obj->priv->gvnc))
return;
- if (priv->absolute) {
+ if (absolute(priv)) {
priv->button_mask = button_mask;
priv->last_x = x;
priv->last_y = y;
@@ -1628,6 +1640,17 @@ static void vnc_display_class_init(VncDisplayClass *klass)
G_PARAM_STATIC_NICK |
G_PARAM_STATIC_BLURB));
g_object_class_install_property (object_class,
+ PROP_POINTER_ALLOW_RELATIVE,
+ g_param_spec_boolean ( "allow-relative-pointer",
+ "Allow Relative Pointer Extension",
+ "Whether we should allow relative mouse reportnig",
+ FALSE,
+ G_PARAM_READWRITE |
+ G_PARAM_CONSTRUCT |
+ G_PARAM_STATIC_NAME |
+ G_PARAM_STATIC_NICK |
+ G_PARAM_STATIC_BLURB));
+ g_object_class_install_property (object_class,
PROP_READ_ONLY,
g_param_spec_boolean ( "read-only",
"Read Only",
@@ -1897,6 +1920,7 @@ static void vnc_display_init(VncDisplay *display)
priv->last_x = -1;
priv->last_y = -1;
priv->absolute = 1;
+ priv->allow_relative = TRUE;
priv->fd = -1;
priv->read_only = FALSE;
priv->allow_lossy = FALSE;
@@ -2042,6 +2066,14 @@ void vnc_display_set_keyboard_grab(VncDisplay *obj, gboolean enable)
do_keyboard_ungrab(obj, FALSE);
}
+void vnc_display_set_pointer_allow_relative(VncDisplay *obj, gboolean enable)
+{
+ VncDisplayPrivate *priv = obj->priv;
+
+ priv->allow_relative = enable;
+ on_pointer_type_change(obj, priv->absolute);
+}
+
void vnc_display_set_read_only(VncDisplay *obj, gboolean enable)
{
obj->priv->read_only = enable;
@@ -2234,7 +2266,7 @@ gboolean vnc_display_get_read_only(VncDisplay *obj)
gboolean vnc_display_is_pointer_absolute(VncDisplay *obj)
{
- return obj->priv->absolute;
+ return absolute(obj->priv);
}
GOptionGroup *
diff --git a/src/vncdisplay.h b/src/vncdisplay.h
index 7c6a64e..2794404 100644
--- a/src/vncdisplay.h
+++ b/src/vncdisplay.h
@@ -106,6 +106,9 @@ gboolean vnc_display_get_pointer_local(VncDisplay *obj);
void vnc_display_set_pointer_grab(VncDisplay *obj, gboolean enable);
gboolean vnc_display_get_pointer_grab(VncDisplay *obj);
+void vnc_display_set_pointer_allow_relative(VncDisplay *obj, gboolean enable);
+gboolean vnc_display_get_pointer_allow_relative(VncDisplay *obj);
+
void vnc_display_set_keyboard_grab(VncDisplay *obj, gboolean enable);
gboolean vnc_display_get_keyboard_grab(VncDisplay *obj);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]