2014-09-11 17:43:32 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <gio/gunixinputstream.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <X11/extensions/sync.h>
|
|
|
|
|
2015-09-23 23:13:59 +00:00
|
|
|
const char *client_id = "0";
|
2014-09-11 17:43:32 +00:00
|
|
|
static gboolean wayland;
|
|
|
|
GHashTable *windows;
|
|
|
|
|
|
|
|
static void read_next_line (GDataInputStream *in);
|
|
|
|
|
|
|
|
static GtkWidget *
|
|
|
|
lookup_window (const char *window_id)
|
|
|
|
{
|
|
|
|
GtkWidget *window = g_hash_table_lookup (windows, window_id);
|
|
|
|
if (!window)
|
|
|
|
g_print ("Window %s doesn't exist", window_id);
|
|
|
|
|
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
2015-09-08 08:03:45 +00:00
|
|
|
static void
|
|
|
|
on_after_paint (GdkFrameClock *clock,
|
|
|
|
GMainLoop *loop)
|
|
|
|
{
|
|
|
|
g_signal_handlers_disconnect_by_func (clock,
|
|
|
|
(gpointer) on_after_paint,
|
|
|
|
loop);
|
|
|
|
g_main_loop_quit (loop);
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
static void
|
|
|
|
process_line (const char *line)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
int argc;
|
|
|
|
char **argv;
|
|
|
|
|
|
|
|
if (!g_shell_parse_argv (line, &argc, &argv, &error))
|
|
|
|
{
|
|
|
|
g_print ("error parsing command: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc < 1)
|
|
|
|
{
|
|
|
|
g_print ("Empty command");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp (argv[0], "create") == 0)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
{
|
2014-09-15 17:24:05 +00:00
|
|
|
g_print ("usage: create <id> [override|csd]");
|
2014-09-11 17:43:32 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_hash_table_lookup (windows, argv[1]))
|
|
|
|
{
|
|
|
|
g_print ("window %s already exists", argv[1]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean override = FALSE;
|
2014-09-15 17:24:05 +00:00
|
|
|
gboolean csd = FALSE;
|
2014-09-11 17:43:32 +00:00
|
|
|
for (i = 2; i < argc; i++)
|
2014-09-15 17:24:05 +00:00
|
|
|
{
|
|
|
|
if (strcmp (argv[i], "override") == 0)
|
|
|
|
override = TRUE;
|
|
|
|
if (strcmp (argv[i], "csd") == 0)
|
|
|
|
csd = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (override && csd)
|
|
|
|
{
|
|
|
|
g_print ("override and csd keywords are exclusie");
|
|
|
|
goto out;
|
|
|
|
}
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
GtkWidget *window = gtk_window_new (override ? GTK_WINDOW_POPUP : GTK_WINDOW_TOPLEVEL);
|
|
|
|
g_hash_table_insert (windows, g_strdup (argv[1]), window);
|
|
|
|
|
2014-09-15 17:24:05 +00:00
|
|
|
if (csd)
|
|
|
|
{
|
|
|
|
GtkWidget *headerbar = gtk_header_bar_new ();
|
|
|
|
gtk_window_set_titlebar (GTK_WINDOW (window), headerbar);
|
|
|
|
gtk_widget_show (headerbar);
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
gtk_window_set_default_size (GTK_WINDOW (window), 100, 100);
|
|
|
|
|
|
|
|
gchar *title = g_strdup_printf ("test/%s/%s", client_id, argv[1]);
|
|
|
|
gtk_window_set_title (GTK_WINDOW (window), title);
|
|
|
|
g_free (title);
|
|
|
|
|
|
|
|
gtk_widget_realize (window);
|
|
|
|
|
|
|
|
if (!wayland)
|
|
|
|
{
|
|
|
|
/* The cairo xlib backend creates a window when initialized, which
|
|
|
|
* confuses our testing if it happens asynchronously the first
|
|
|
|
* time a window is painted. By creating an Xlib surface and
|
|
|
|
* destroying it, we force initialization at a more predictable time.
|
|
|
|
*/
|
|
|
|
GdkWindow *window_gdk = gtk_widget_get_window (window);
|
|
|
|
cairo_surface_t *surface = gdk_window_create_similar_surface (window_gdk,
|
|
|
|
CAIRO_CONTENT_COLOR,
|
|
|
|
1, 1);
|
|
|
|
cairo_surface_destroy (surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2015-09-24 05:39:58 +00:00
|
|
|
else if (strcmp (argv[0], "set_parent") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 3)
|
|
|
|
{
|
|
|
|
g_print ("usage: menu <window-id> <parent-id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
{
|
|
|
|
g_print ("unknown window %s", argv[1]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *parent_window = lookup_window (argv[2]);
|
|
|
|
if (!parent_window)
|
|
|
|
{
|
|
|
|
g_print ("unknown parent window %s", argv[2]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (window),
|
|
|
|
GTK_WINDOW (parent_window));
|
|
|
|
}
|
2014-09-11 17:43:32 +00:00
|
|
|
else if (strcmp (argv[0], "show") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: show <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
2015-09-08 08:03:45 +00:00
|
|
|
GdkWindow *gdk_window = gtk_widget_get_window (window);
|
2014-09-11 17:43:32 +00:00
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gtk_widget_show (window);
|
2015-09-08 08:03:45 +00:00
|
|
|
|
|
|
|
/* When a Wayland client, we cannot be really sure that the window has
|
|
|
|
* been mappable until after we have painted. So, in order to have the
|
|
|
|
* test runner rely on the "show" command to have done what the client
|
|
|
|
* needs to do in order for a window to be mappable compositor side, lets
|
|
|
|
* wait with returning until after the first frame.
|
|
|
|
*/
|
|
|
|
GdkFrameClock *frame_clock = gdk_window_get_frame_clock (gdk_window);
|
|
|
|
GMainLoop *loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
g_signal_connect (frame_clock, "after-paint",
|
|
|
|
G_CALLBACK (on_after_paint),
|
|
|
|
loop);
|
|
|
|
g_main_loop_run (loop);
|
|
|
|
g_main_loop_unref (loop);
|
2014-09-11 17:43:32 +00:00
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "hide") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: hide <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gtk_widget_hide (window);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "activate") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: activate <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gtk_window_present (GTK_WINDOW (window));
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "raise") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: raise <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gdk_window_raise (gtk_widget_get_window (window));
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "lower") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: lower <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gdk_window_lower (gtk_widget_get_window (window));
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "destroy") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: destroy <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
g_hash_table_remove (windows, argv[1]);
|
|
|
|
gtk_widget_destroy (window);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "destroy_all") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 1)
|
|
|
|
{
|
|
|
|
g_print ("usage: destroy_all");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer key, value;
|
|
|
|
|
|
|
|
g_hash_table_iter_init (&iter, windows);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
gtk_widget_destroy (value);
|
|
|
|
|
|
|
|
g_hash_table_remove_all (windows);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "sync") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 1)
|
|
|
|
{
|
|
|
|
g_print ("usage: sync");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
gdk_display_sync (gdk_display_get_default ());
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "set_counter") == 0)
|
|
|
|
{
|
|
|
|
XSyncCounter counter;
|
|
|
|
int value;
|
|
|
|
|
|
|
|
if (argc != 3)
|
|
|
|
{
|
|
|
|
g_print ("usage: set_counter <counter> <value>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wayland)
|
|
|
|
{
|
|
|
|
g_print ("usage: set_counter can only be used for X11");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = strtoul(argv[1], NULL, 10);
|
|
|
|
value = atoi(argv[2]);
|
|
|
|
XSyncValue sync_value;
|
|
|
|
XSyncIntToValue (&sync_value, value);
|
|
|
|
|
|
|
|
XSyncSetCounter (gdk_x11_display_get_xdisplay (gdk_display_get_default ()),
|
|
|
|
counter, sync_value);
|
|
|
|
}
|
2014-09-24 20:09:20 +00:00
|
|
|
else if (strcmp (argv[0], "minimize") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: minimize <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gtk_window_iconify (GTK_WINDOW (window));
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "unminimize") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
{
|
|
|
|
g_print ("usage: unminimize <id>");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
GtkWidget *window = lookup_window (argv[1]);
|
|
|
|
if (!window)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
gtk_window_deiconify (GTK_WINDOW (window));
|
|
|
|
}
|
2014-09-11 17:43:32 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
g_print ("Unknown command %s", argv[0]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_print ("OK\n");
|
|
|
|
|
|
|
|
out:
|
|
|
|
g_strfreev (argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_line_received (GObject *source,
|
|
|
|
GAsyncResult *result,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GDataInputStream *in = G_DATA_INPUT_STREAM (source);
|
|
|
|
GError *error = NULL;
|
|
|
|
gsize length;
|
|
|
|
char *line = g_data_input_stream_read_line_finish_utf8 (in, result, &length, &error);
|
|
|
|
|
|
|
|
if (line == NULL)
|
|
|
|
{
|
|
|
|
if (error != NULL)
|
|
|
|
g_printerr ("Error reading from stdin: %s\n", error->message);
|
|
|
|
gtk_main_quit ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
process_line (line);
|
|
|
|
g_free (line);
|
|
|
|
read_next_line (in);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
read_next_line (GDataInputStream *in)
|
|
|
|
{
|
|
|
|
g_data_input_stream_read_line_async (in, G_PRIORITY_DEFAULT, NULL,
|
|
|
|
on_line_received, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
const GOptionEntry options[] = {
|
|
|
|
{
|
|
|
|
"wayland", 0, 0, G_OPTION_ARG_NONE,
|
|
|
|
&wayland,
|
|
|
|
"Create a wayland client, not an X11 one",
|
|
|
|
NULL
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"client-id", 0, 0, G_OPTION_ARG_STRING,
|
|
|
|
&client_id,
|
|
|
|
"Identifier used in Window titles for this client",
|
|
|
|
"CLIENT_ID",
|
|
|
|
},
|
|
|
|
{ NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
GOptionContext *context = g_option_context_new (NULL);
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
g_option_context_add_main_entries (context, options, NULL);
|
|
|
|
|
|
|
|
if (!g_option_context_parse (context,
|
|
|
|
&argc, &argv, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wayland)
|
|
|
|
gdk_set_allowed_backends ("wayland");
|
|
|
|
else
|
|
|
|
gdk_set_allowed_backends ("x11");
|
|
|
|
|
|
|
|
gtk_init (NULL, NULL);
|
|
|
|
|
|
|
|
windows = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
|
|
g_free, NULL);
|
|
|
|
|
|
|
|
GInputStream *raw_in = g_unix_input_stream_new (0, FALSE);
|
|
|
|
GDataInputStream *in = g_data_input_stream_new (raw_in);
|
|
|
|
|
|
|
|
read_next_line (in);
|
|
|
|
|
|
|
|
gtk_main ();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|