forked from AuroraMiddleware/gtk
08f57d5c3d
Implement the selection interface for GtkListBox. This also includes a convenience api for context addresses: gtk_at_spi_context_to_ref.
260 lines
7.8 KiB
C
260 lines
7.8 KiB
C
/* gtkatspiselection.c: AT-SPI Selection 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 "gtkatspiselectionprivate.h"
|
|
|
|
#include "a11y/atspi/atspi-selection.h"
|
|
|
|
#include "gtkatcontextprivate.h"
|
|
#include "gtkatspicontextprivate.h"
|
|
#include "gtkaccessibleprivate.h"
|
|
#include "gtkdebug.h"
|
|
#include "gtklistbox.h"
|
|
|
|
#include <gio/gio.h>
|
|
|
|
typedef struct {
|
|
int n;
|
|
GtkListBoxRow *row;
|
|
} RowCounter;
|
|
|
|
static void
|
|
find_nth (GtkListBox *box,
|
|
GtkListBoxRow *row,
|
|
gpointer data)
|
|
{
|
|
RowCounter *counter = data;
|
|
|
|
if (counter->n == 0)
|
|
counter->row = row;
|
|
|
|
counter->n--;
|
|
}
|
|
|
|
static void
|
|
listbox_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, "GetSelectedChild") == 0)
|
|
{
|
|
RowCounter counter;
|
|
int idx;
|
|
|
|
g_variant_get (parameters, "(i)", &idx);
|
|
|
|
counter.n = idx;
|
|
counter.row = NULL;
|
|
|
|
gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), find_nth, &counter);
|
|
|
|
if (counter.row == NULL)
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No selected child for %d", idx);
|
|
else
|
|
{
|
|
GtkATContext *ctx = gtk_accessible_get_at_context (GTK_ACCESSIBLE (counter.row));
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(@(so))", gtk_at_spi_context_to_ref (GTK_AT_SPI_CONTEXT (ctx))));
|
|
}
|
|
}
|
|
else if (g_strcmp0 (method_name, "SelectChild") == 0)
|
|
{
|
|
int idx;
|
|
GtkListBoxRow *row;
|
|
|
|
g_variant_get (parameters, "(i)", &idx);
|
|
|
|
row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
|
|
if (!row)
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
|
|
else
|
|
{
|
|
gboolean ret;
|
|
|
|
gtk_list_box_select_row (GTK_LIST_BOX (widget), row);
|
|
ret = gtk_list_box_row_is_selected (row);
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
|
|
}
|
|
}
|
|
else if (g_strcmp0 (method_name, "DeselectChild") == 0)
|
|
{
|
|
int idx;
|
|
GtkListBoxRow *row;
|
|
|
|
g_variant_get (parameters, "(i)", &idx);
|
|
|
|
row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
|
|
if (!row)
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
|
|
else
|
|
{
|
|
gboolean ret;
|
|
|
|
gtk_list_box_unselect_row (GTK_LIST_BOX (widget), row);
|
|
ret = !gtk_list_box_row_is_selected (row);
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
|
|
}
|
|
}
|
|
else if (g_strcmp0 (method_name, "DeselectSelectedChild") == 0)
|
|
{
|
|
RowCounter counter;
|
|
int idx;
|
|
|
|
g_variant_get (parameters, "(i)", &idx);
|
|
|
|
counter.n = idx;
|
|
counter.row = NULL;
|
|
|
|
gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), find_nth, &counter);
|
|
|
|
if (counter.row == NULL)
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No selected child for %d", idx);
|
|
else
|
|
{
|
|
gboolean ret;
|
|
|
|
gtk_list_box_unselect_row (GTK_LIST_BOX (widget), counter.row);
|
|
ret = !gtk_list_box_row_is_selected (counter.row);
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret));
|
|
}
|
|
}
|
|
else if (g_strcmp0 (method_name, "IsChildSelected") == 0)
|
|
{
|
|
int idx;
|
|
GtkListBoxRow *row;
|
|
|
|
g_variant_get (parameters, "(i)", &idx);
|
|
|
|
row = gtk_list_box_get_row_at_index (GTK_LIST_BOX (widget), idx);
|
|
if (!row)
|
|
g_dbus_method_invocation_return_error (invocation, G_DBUS_ERROR, G_DBUS_ERROR_INVALID_ARGS, "No child at position %d", idx);
|
|
else
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", gtk_list_box_row_is_selected (row)));
|
|
}
|
|
else if (g_strcmp0 (method_name, "SelectAll") == 0)
|
|
{
|
|
gtk_list_box_select_all (GTK_LIST_BOX (widget));
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", TRUE));
|
|
}
|
|
else if (g_strcmp0 (method_name, "ClearSelection") == 0)
|
|
{
|
|
gtk_list_box_unselect_all (GTK_LIST_BOX (widget));
|
|
g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", TRUE));
|
|
}
|
|
}
|
|
|
|
static void
|
|
count_selected (GtkListBox *box,
|
|
GtkListBoxRow *row,
|
|
gpointer data)
|
|
{
|
|
*(int *)data += 1;
|
|
}
|
|
|
|
static GVariant *
|
|
listbox_get_property (GDBusConnection *connection,
|
|
const gchar *sender,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *property_name,
|
|
GError **error,
|
|
gpointer user_data)
|
|
{
|
|
GtkATContext *self = GTK_AT_CONTEXT (user_data);
|
|
GtkAccessible *accessible = gtk_at_context_get_accessible (self);
|
|
GtkWidget *widget = GTK_WIDGET (accessible);
|
|
|
|
if (g_strcmp0 (property_name, "NSelectedChildren") == 0)
|
|
{
|
|
int count = 0;
|
|
|
|
gtk_list_box_selected_foreach (GTK_LIST_BOX (widget), count_selected, &count);
|
|
|
|
return g_variant_new_int32 (count);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static const GDBusInterfaceVTable listbox_vtable = {
|
|
listbox_handle_method,
|
|
listbox_get_property,
|
|
NULL
|
|
};
|
|
|
|
const GDBusInterfaceVTable *
|
|
gtk_atspi_get_selection_vtable (GtkWidget *widget)
|
|
{
|
|
if (GTK_IS_LIST_BOX(widget))
|
|
return &listbox_vtable;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
typedef struct {
|
|
GtkAtspiSelectionCallback *changed;
|
|
gpointer data;
|
|
} SelectionChanged;
|
|
|
|
void
|
|
gtk_atspi_connect_selection_signals (GtkWidget *widget,
|
|
GtkAtspiSelectionCallback selection_changed,
|
|
gpointer data)
|
|
{
|
|
if (GTK_IS_LIST_BOX (widget))
|
|
{
|
|
SelectionChanged *changed;
|
|
|
|
changed = g_new (SelectionChanged, 1);
|
|
changed->changed = selection_changed;
|
|
changed->data = data;
|
|
|
|
g_object_set_data_full (G_OBJECT (widget), "accessible-selection-data", changed, g_free);
|
|
|
|
g_signal_connect_swapped (widget, "selected-rows-changed", G_CALLBACK (selection_changed), data);
|
|
}
|
|
}
|
|
|
|
void
|
|
gtk_atspi_disconnect_selection_signals (GtkWidget *widget)
|
|
{
|
|
if (GTK_IS_LIST_BOX (widget))
|
|
{
|
|
SelectionChanged *changed;
|
|
|
|
changed = g_object_get_data (G_OBJECT (widget), "accessible-selection-data");
|
|
|
|
g_signal_handlers_disconnect_by_func (widget, changed->changed, changed->data);
|
|
|
|
g_object_set_data (G_OBJECT (widget), "accessible-selection-data", NULL);
|
|
}
|
|
}
|
|
|