256 lines
7.9 KiB
C
256 lines
7.9 KiB
C
/* gtkatspicomponent.c: AT-SPI Component implementation
|
|
*
|
|
* Copyright 2020 Red Hat, Inc.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*
|
|
* 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.1 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
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkatspicomponentprivate.h"
|
|
|
|
#include "gtkatspicontextprivate.h"
|
|
#include "gtkatspiprivate.h"
|
|
#include "gtkatspiutilsprivate.h"
|
|
#include "gtkaccessibleprivate.h"
|
|
#include "gtkpopover.h"
|
|
#include "gtkwidget.h"
|
|
#include "gtkwindow.h"
|
|
|
|
#include "a11y/atspi/atspi-component.h"
|
|
|
|
#include "gtkdebug.h"
|
|
|
|
#include <gio/gio.h>
|
|
|
|
static void
|
|
translate_coordinates_to_widget (GtkWidget *widget,
|
|
AtspiCoordType coordtype,
|
|
int xi,
|
|
int yi,
|
|
int *xo,
|
|
int *yo)
|
|
{
|
|
graphene_point_t p;
|
|
GtkWidget *source;
|
|
|
|
switch (coordtype)
|
|
{
|
|
case ATSPI_COORD_TYPE_SCREEN:
|
|
*xo = 0;
|
|
*yo = 0;
|
|
return;
|
|
|
|
case ATSPI_COORD_TYPE_WINDOW:
|
|
source = GTK_WIDGET (gtk_widget_get_root (widget));
|
|
break;
|
|
|
|
case ATSPI_COORD_TYPE_PARENT:
|
|
source = gtk_widget_get_parent (widget);
|
|
break;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
if (!gtk_widget_compute_point (source, widget, &GRAPHENE_POINT_INIT (xi, yi), &p))
|
|
graphene_point_init (&p, xi, yi);
|
|
|
|
*xo = (int)p.x;
|
|
*yo = (int)p.y;
|
|
}
|
|
|
|
static void
|
|
translate_coordinates_from_widget (GtkWidget *widget,
|
|
AtspiCoordType coordtype,
|
|
int xi,
|
|
int yi,
|
|
int *xo,
|
|
int *yo)
|
|
{
|
|
graphene_point_t p;
|
|
GtkWidget *target;
|
|
|
|
switch (coordtype)
|
|
{
|
|
case ATSPI_COORD_TYPE_SCREEN:
|
|
*xo = 0;
|
|
*yo = 0;
|
|
return;
|
|
|
|
case ATSPI_COORD_TYPE_WINDOW:
|
|
target = GTK_WIDGET (gtk_widget_get_root (widget));
|
|
break;
|
|
|
|
case ATSPI_COORD_TYPE_PARENT:
|
|
target = gtk_widget_get_parent (widget);
|
|
break;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
}
|
|
|
|
if (!gtk_widget_compute_point (widget, target, &GRAPHENE_POINT_INIT (xi, yi), &p))
|
|
graphene_point_init (&p, xi, yi);
|
|
|
|
*xo = (int)p.x;
|
|
*yo = (int)p.y;
|
|
}
|
|
|
|
static void
|
|
component_handle_method (GDBusConnection *connection,
|
|
const gchar *sender,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *method_name,
|
|
GVariant *parameters,
|
|
GDBusMethodInvocation *invocation,
|
|
gpointer user_data)
|
|
{
|
|
GtkATContext *self = user_data;
|
|
GtkAccessible *accessible = gtk_at_context_get_accessible (self);
|
|
GtkWidget *widget = GTK_WIDGET (accessible);
|
|
|
|
if (g_strcmp0 (method_name, "Contains") == 0)
|
|
{
|
|
int x, y;
|
|
AtspiCoordType coordtype;
|
|
gboolean ret;
|
|
|
|
g_variant_get (parameters, "(iiu)", &x, &y, &coordtype);
|
|
|
|
translate_coordinates_to_widget (widget, coordtype, x, y, &x, &y);
|
|
|
|
ret = gtk_widget_contains (widget, x, y);
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetAccessibleAtPoint") == 0)
|
|
{
|
|
int x, y;
|
|
AtspiCoordType coordtype;
|
|
GtkWidget *child;
|
|
|
|
g_variant_get (parameters, "(iiu)", &x, &y, &coordtype);
|
|
|
|
translate_coordinates_to_widget (widget, coordtype, x, y, &x, &y);
|
|
|
|
child = gtk_widget_pick (widget, x, y, GTK_PICK_DEFAULT);
|
|
if (!child)
|
|
{
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(@(so))", gtk_at_spi_null_ref ()));
|
|
}
|
|
else
|
|
{
|
|
GtkATContext *context = gtk_accessible_get_at_context (GTK_ACCESSIBLE (child));
|
|
GtkAtSpiContext *ctx = GTK_AT_SPI_CONTEXT (context);
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(@(so))", gtk_at_spi_context_to_ref (ctx)));
|
|
|
|
g_object_unref (context);
|
|
}
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetExtents") == 0)
|
|
{
|
|
AtspiCoordType coordtype;
|
|
int x, y;
|
|
int width = gtk_widget_get_width (widget);
|
|
int height = gtk_widget_get_height (widget);
|
|
|
|
g_variant_get (parameters, "(u)", &coordtype);
|
|
|
|
translate_coordinates_from_widget (widget, coordtype, 0, 0, &x, &y);
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("((iiii))", x, y, width, height));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetPosition") == 0)
|
|
{
|
|
AtspiCoordType coordtype;
|
|
int x, y;
|
|
|
|
g_variant_get (parameters, "(u)", &coordtype);
|
|
|
|
translate_coordinates_from_widget (widget, coordtype, 0, 0, &x, &y);
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(ii)", x, y));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetSize") == 0)
|
|
{
|
|
int width = gtk_widget_get_width (widget);
|
|
int height = gtk_widget_get_height (widget);
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(ii)", width, height));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetLayer") == 0)
|
|
{
|
|
AtspiComponentLayer layer;
|
|
|
|
if (GTK_IS_WINDOW (widget))
|
|
layer = ATSPI_COMPONENT_LAYER_WINDOW;
|
|
else if (GTK_IS_POPOVER (widget))
|
|
layer = ATSPI_COMPONENT_LAYER_POPUP;
|
|
else
|
|
layer = ATSPI_COMPONENT_LAYER_WIDGET;
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(u)", layer));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetMDIZOrder") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(n)", 0));
|
|
}
|
|
else if (g_strcmp0 (method_name, "GrabFocus") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
else if (g_strcmp0 (method_name, "GetAlpha") == 0)
|
|
{
|
|
double opacity = gtk_widget_get_opacity (widget);
|
|
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(d)", opacity));
|
|
}
|
|
else if (g_strcmp0 (method_name, "SetExtents") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
else if (g_strcmp0 (method_name, "SetPosition") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
else if (g_strcmp0 (method_name, "SetSize") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
else if (g_strcmp0 (method_name, "ScrollTo") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
else if (g_strcmp0 (method_name, "ScrollToPoint") == 0)
|
|
{
|
|
g_dbus_method_invocation_return_error_literal (invocation, G_DBUS_ERROR, G_DBUS_ERROR_NOT_SUPPORTED, "");
|
|
}
|
|
}
|
|
|
|
static const GDBusInterfaceVTable component_vtable = {
|
|
component_handle_method,
|
|
};
|
|
|
|
const GDBusInterfaceVTable *
|
|
gtk_atspi_get_component_vtable (GtkAccessible *accessible)
|
|
{
|
|
if (GTK_IS_WIDGET (accessible))
|
|
return &component_vtable;
|
|
|
|
return NULL;
|
|
}
|