gtk: misc grab tweaks, locale fix.

-----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2.0.22 (GNU/Linux)
 
 iQIcBAABAgAGBQJV9/R4AAoJEEy22O7T6HE4gPkP/R3+vDFsIf2uUl6R/XDFBvOc
 0wF9Ft+8I2XJxOGdLMm49m/tJoohYzi3EnOA7yeQa9b63C56DzMawaf30F2Si0X/
 lVv80HfmGsD0d/TZVuJxHswkSu+BiHnpq8pcefvGo19fb3niRa5fwpVHf6y8woWZ
 8QXdL9d89e+hHe1YRXbDleEGasCUk47bJEtKqi7h9qvcvSgGWRf+zGWHFWc0JH/Q
 5TPV40Yp9G6E8E5/9VxR3FCLnv4rxgRqtzPsZX+3M4w31HfFxLogC6LStoAFUwac
 RnQrLA/NW7PeQUx0Sct5Aqsaw2VhOwWp1NSTKjdXJFySxT7iCfKfYsNkpZynYT2U
 +ODjvuZMUT0tBWZUZINjLw0DXnGfRGc9prMvtqKGPZ5D+kfUaLR9zDbu/BwcnfYB
 LJ3LCqsA2IlyA+UnLqo7xnG5ba4ceyZpF6tpXhfBQAlf0l9Gw4wbCjBnaBxLO4Ib
 hKYKvryQNL/j+tcBU8/ImrVbdUHWljndvukjP/TOCX59UWarryS4oNPw1WVp3IBv
 CGNwgboq+QNZ2aIhXuOtDCJ3K/9F5AI9pX79nZ9jmxovCHpFnEa5zVBQ33xubyr0
 n4JzGC+CWM9vwhkhsm3huRyHbAjdDqSJ1FRuN8z+LHL6XMvhS9zhUmjgJVyGoe9M
 9jVVwrLxvL52DFYLPz5V
 =ANFa
 -----END PGP SIGNATURE-----

Merge remote-tracking branch 'remotes/kraxel/tags/pull-gtk-20150915-1' into staging

gtk: misc grab tweaks, locale fix.

# gpg: Signature made Tue 15 Sep 2015 11:35:36 BST using RSA key ID D3E87138
# gpg: Good signature from "Gerd Hoffmann (work) <kraxel@redhat.com>"
# gpg:                 aka "Gerd Hoffmann <gerd@kraxel.org>"
# gpg:                 aka "Gerd Hoffmann (private) <kraxel@gmail.com>"

* remotes/kraxel/tags/pull-gtk-20150915-1:
  gtk: use setlocale() for LC_MESSAGES only
  gtk: don't grab input when entering fullscreen.
  gtk: set free_scale when setting zoom_fit
  gtk: trace input grab reason
  gtk: move gd_update_caption calls to gd_{grab,ungrab}_{pointer,keyboard}
  gtk: check for existing grabs in gd_grab_{pointer,keyboard}

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2015-09-15 14:11:28 +01:00
commit 1078f5db8a
2 changed files with 58 additions and 29 deletions

View File

@ -1150,7 +1150,8 @@ ppm_save(const char *filename, void *display_surface) "%s surface=%p"
gd_switch(const char *tab, int width, int height) "tab=%s, width=%d, height=%d" gd_switch(const char *tab, int width, int height) "tab=%s, width=%d, height=%d"
gd_update(const char *tab, int x, int y, int w, int h) "tab=%s, x=%d, y=%d, w=%d, h=%d" gd_update(const char *tab, int x, int y, int w, int h) "tab=%s, x=%d, y=%d, w=%d, h=%d"
gd_key_event(const char *tab, int gdk_keycode, int qemu_keycode, const char *action) "tab=%s, translated GDK keycode %d to QEMU keycode %d (%s)" gd_key_event(const char *tab, int gdk_keycode, int qemu_keycode, const char *action) "tab=%s, translated GDK keycode %d to QEMU keycode %d (%s)"
gd_grab(const char *tab, const char *device, bool on) "tab=%s, %s %d" gd_grab(const char *tab, const char *device, const char *reason) "tab=%s, dev=%s, reason=%s"
gd_ungrab(const char *tab, const char *device) "tab=%s, dev=%s"
# ui/vnc.c # ui/vnc.c
vnc_key_guest_leds(bool caps, bool num, bool scroll) "caps %d, num %d, scroll %d" vnc_key_guest_leds(bool caps, bool num, bool scroll) "caps %d, num %d, scroll %d"

View File

@ -165,8 +165,10 @@ struct GtkDisplayState {
bool ignore_keys; bool ignore_keys;
}; };
static void gd_grab_pointer(VirtualConsole *vc); static void gd_grab_pointer(VirtualConsole *vc, const char *reason);
static void gd_ungrab_pointer(GtkDisplayState *s); static void gd_ungrab_pointer(GtkDisplayState *s);
static void gd_grab_keyboard(VirtualConsole *vc, const char *reason);
static void gd_ungrab_keyboard(GtkDisplayState *s);
/** Utility Functions **/ /** Utility Functions **/
@ -849,13 +851,11 @@ static gboolean gd_button_event(GtkWidget *widget, GdkEventButton *button,
/* implicitly grab the input at the first click in the relative mode */ /* implicitly grab the input at the first click in the relative mode */
if (button->button == 1 && button->type == GDK_BUTTON_PRESS && if (button->button == 1 && button->type == GDK_BUTTON_PRESS &&
!qemu_input_is_absolute() && s->ptr_owner != vc) { !qemu_input_is_absolute() && s->ptr_owner != vc) {
gd_ungrab_pointer(s);
if (!vc->window) { if (!vc->window) {
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(s->grab_item), gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(s->grab_item),
TRUE); TRUE);
} else { } else {
gd_grab_pointer(vc); gd_grab_pointer(vc, "relative-mode-click");
gd_update_caption(s);
} }
return TRUE; return TRUE;
} }
@ -1092,9 +1092,8 @@ static gboolean gd_win_grab(void *opaque)
if (vc->s->ptr_owner) { if (vc->s->ptr_owner) {
gd_ungrab_pointer(vc->s); gd_ungrab_pointer(vc->s);
} else { } else {
gd_grab_pointer(vc); gd_grab_pointer(vc, "user-request-detached-tab");
} }
gd_update_caption(vc->s);
return TRUE; return TRUE;
} }
@ -1141,10 +1140,6 @@ static void gd_menu_full_screen(GtkMenuItem *item, void *opaque)
gtk_widget_hide(s->menu_bar); gtk_widget_hide(s->menu_bar);
if (vc->type == GD_VC_GFX) { if (vc->type == GD_VC_GFX) {
gtk_widget_set_size_request(vc->gfx.drawing_area, -1, -1); gtk_widget_set_size_request(vc->gfx.drawing_area, -1, -1);
if (qemu_console_is_graphic(vc->gfx.dcl.con)) {
gtk_check_menu_item_set_active
(GTK_CHECK_MENU_ITEM(s->grab_item), TRUE);
}
} }
gtk_window_fullscreen(GTK_WINDOW(s->window)); gtk_window_fullscreen(GTK_WINDOW(s->window));
s->full_screen = TRUE; s->full_screen = TRUE;
@ -1157,8 +1152,6 @@ static void gd_menu_full_screen(GtkMenuItem *item, void *opaque)
vc->gfx.scale_x = 1.0; vc->gfx.scale_x = 1.0;
vc->gfx.scale_y = 1.0; vc->gfx.scale_y = 1.0;
gd_update_windowsize(vc); gd_update_windowsize(vc);
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(s->grab_item),
FALSE);
} }
} }
@ -1257,8 +1250,16 @@ static void gd_grab_devices(VirtualConsole *vc, bool grab,
} }
#endif #endif
static void gd_grab_keyboard(VirtualConsole *vc) static void gd_grab_keyboard(VirtualConsole *vc, const char *reason)
{ {
if (vc->s->kbd_owner) {
if (vc->s->kbd_owner == vc) {
return;
} else {
gd_ungrab_keyboard(vc->s);
}
}
#if GTK_CHECK_VERSION(3, 0, 0) #if GTK_CHECK_VERSION(3, 0, 0)
gd_grab_devices(vc, true, GDK_SOURCE_KEYBOARD, gd_grab_devices(vc, true, GDK_SOURCE_KEYBOARD,
GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK, GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK,
@ -1269,7 +1270,8 @@ static void gd_grab_keyboard(VirtualConsole *vc)
GDK_CURRENT_TIME); GDK_CURRENT_TIME);
#endif #endif
vc->s->kbd_owner = vc; vc->s->kbd_owner = vc;
trace_gd_grab(vc->label, "kbd", true); gd_update_caption(vc->s);
trace_gd_grab(vc->label, "kbd", reason);
} }
static void gd_ungrab_keyboard(GtkDisplayState *s) static void gd_ungrab_keyboard(GtkDisplayState *s)
@ -1286,12 +1288,22 @@ static void gd_ungrab_keyboard(GtkDisplayState *s)
#else #else
gdk_keyboard_ungrab(GDK_CURRENT_TIME); gdk_keyboard_ungrab(GDK_CURRENT_TIME);
#endif #endif
trace_gd_grab(vc->label, "kbd", false); gd_update_caption(s);
trace_gd_ungrab(vc->label, "kbd");
} }
static void gd_grab_pointer(VirtualConsole *vc) static void gd_grab_pointer(VirtualConsole *vc, const char *reason)
{ {
GdkDisplay *display = gtk_widget_get_display(vc->gfx.drawing_area); GdkDisplay *display = gtk_widget_get_display(vc->gfx.drawing_area);
if (vc->s->ptr_owner) {
if (vc->s->ptr_owner == vc) {
return;
} else {
gd_ungrab_pointer(vc->s);
}
}
#if GTK_CHECK_VERSION(3, 0, 0) #if GTK_CHECK_VERSION(3, 0, 0)
GdkDeviceManager *mgr = gdk_display_get_device_manager(display); GdkDeviceManager *mgr = gdk_display_get_device_manager(display);
gd_grab_devices(vc, true, GDK_SOURCE_MOUSE, gd_grab_devices(vc, true, GDK_SOURCE_MOUSE,
@ -1318,7 +1330,8 @@ static void gd_grab_pointer(VirtualConsole *vc)
&vc->s->grab_x_root, &vc->s->grab_y_root, NULL); &vc->s->grab_x_root, &vc->s->grab_y_root, NULL);
#endif #endif
vc->s->ptr_owner = vc; vc->s->ptr_owner = vc;
trace_gd_grab(vc->label, "ptr", true); gd_update_caption(vc->s);
trace_gd_grab(vc->label, "ptr", reason);
} }
static void gd_ungrab_pointer(GtkDisplayState *s) static void gd_ungrab_pointer(GtkDisplayState *s)
@ -1343,7 +1356,8 @@ static void gd_ungrab_pointer(GtkDisplayState *s)
gtk_widget_get_screen(vc->gfx.drawing_area), gtk_widget_get_screen(vc->gfx.drawing_area),
vc->s->grab_x_root, vc->s->grab_y_root); vc->s->grab_x_root, vc->s->grab_y_root);
#endif #endif
trace_gd_grab(vc->label, "ptr", false); gd_update_caption(s);
trace_gd_ungrab(vc->label, "ptr");
} }
static void gd_menu_grab_input(GtkMenuItem *item, void *opaque) static void gd_menu_grab_input(GtkMenuItem *item, void *opaque)
@ -1352,16 +1366,13 @@ static void gd_menu_grab_input(GtkMenuItem *item, void *opaque)
VirtualConsole *vc = gd_vc_find_current(s); VirtualConsole *vc = gd_vc_find_current(s);
if (gd_is_grab_active(s)) { if (gd_is_grab_active(s)) {
if (!gd_grab_on_hover(s)) { gd_grab_keyboard(vc, "user-request-main-window");
gd_grab_keyboard(vc); gd_grab_pointer(vc, "user-request-main-window");
}
gd_grab_pointer(vc);
} else { } else {
gd_ungrab_keyboard(s); gd_ungrab_keyboard(s);
gd_ungrab_pointer(s); gd_ungrab_pointer(s);
} }
gd_update_caption(s);
gd_update_cursor(vc); gd_update_cursor(vc);
} }
@ -1415,9 +1426,7 @@ static gboolean gd_enter_event(GtkWidget *widget, GdkEventCrossing *crossing,
GtkDisplayState *s = vc->s; GtkDisplayState *s = vc->s;
if (gd_grab_on_hover(s)) { if (gd_grab_on_hover(s)) {
gd_ungrab_keyboard(s); gd_grab_keyboard(vc, "grab-on-hover");
gd_grab_keyboard(vc);
gd_update_caption(s);
} }
return TRUE; return TRUE;
} }
@ -1430,7 +1439,6 @@ static gboolean gd_leave_event(GtkWidget *widget, GdkEventCrossing *crossing,
if (gd_grab_on_hover(s)) { if (gd_grab_on_hover(s)) {
gd_ungrab_keyboard(s); gd_ungrab_keyboard(s);
gd_update_caption(s);
} }
return TRUE; return TRUE;
} }
@ -1768,6 +1776,7 @@ static GSList *gd_vc_gfx_init(GtkDisplayState *s, VirtualConsole *vc,
if (dpy_ui_info_supported(vc->gfx.dcl.con)) { if (dpy_ui_info_supported(vc->gfx.dcl.con)) {
gtk_menu_item_activate(GTK_MENU_ITEM(s->zoom_fit_item)); gtk_menu_item_activate(GTK_MENU_ITEM(s->zoom_fit_item));
s->free_scale = true;
} }
return group; return group;
@ -1941,7 +1950,8 @@ void gtk_display_init(DisplayState *ds, bool full_screen, bool grab_on_hover)
s->free_scale = FALSE; s->free_scale = FALSE;
setlocale(LC_ALL, ""); /* LC_MESSAGES only. See early_gtk_display_init() for details */
setlocale(LC_MESSAGES, "");
bindtextdomain("qemu", CONFIG_QEMU_LOCALEDIR); bindtextdomain("qemu", CONFIG_QEMU_LOCALEDIR);
textdomain("qemu"); textdomain("qemu");
@ -2010,6 +2020,24 @@ void gtk_display_init(DisplayState *ds, bool full_screen, bool grab_on_hover)
void early_gtk_display_init(int opengl) void early_gtk_display_init(int opengl)
{ {
/* The QEMU code relies on the assumption that it's always run in
* the C locale. Therefore it is not prepared to deal with
* operations that produce different results depending on the
* locale, such as printf's formatting of decimal numbers, and
* possibly others.
*
* Since GTK+ calls setlocale() by default -importing the locale
* settings from the environment- we must prevent it from doing so
* using gtk_disable_setlocale().
*
* QEMU's GTK+ UI, however, _does_ have translations for some of
* the menu items. As a trade-off between a functionally correct
* QEMU and a fully internationalized UI we support importing
* LC_MESSAGES from the environment (see the setlocale() call
* earlier in this file). This allows us to display translated
* messages leaving everything else untouched.
*/
gtk_disable_setlocale();
gtkinit = gtk_init_check(NULL, NULL); gtkinit = gtk_init_check(NULL, NULL);
if (!gtkinit) { if (!gtkinit) {
/* don't exit yet, that'll break -help */ /* don't exit yet, that'll break -help */