/* gtkatspieditabletext.c: EditableText interface for GtkAtspiContext * * 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 . */ #include "config.h" #include "gtkatspieditabletextprivate.h" #include "gtkatcontextprivate.h" #include "a11y/atspi/atspi-editabletext.h" #include "gtkeditable.h" #include "gtkentry.h" #include "gtksearchentry.h" #include "gtkpasswordentry.h" #include "gtkspinbutton.h" #include "gtktextview.h" #include /* {{{ GtkEditable */ typedef struct { GtkWidget *widget; int position; } PasteData; static void text_received (GObject *source, GAsyncResult *result, gpointer data) { GdkClipboard *clipboard = GDK_CLIPBOARD (source); PasteData *pdata = data; char *text; text = gdk_clipboard_read_text_finish (clipboard, result, NULL); if (text) gtk_editable_insert_text (GTK_EDITABLE (pdata->widget), text, -1, &pdata->position); g_free (text); g_free (pdata); } static void entry_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, "SetTextContents") == 0) { char *text; gboolean ret = FALSE; g_variant_get (parameters, "(&s)", &text); if (gtk_editable_get_editable (GTK_EDITABLE (widget))) { gtk_editable_set_text (GTK_EDITABLE (widget), text); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "InsertText") == 0) { int position; char *text; int len; gboolean ret = FALSE; g_variant_get (parameters, "(i&si)", &position, &text, &len); if (gtk_editable_get_editable (GTK_EDITABLE (widget))) { gtk_editable_insert_text (GTK_EDITABLE (widget), text, -1, &position); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "CopyText") == 0) { int start, end; char *str; g_variant_get (parameters, "(ii)", &start, &end); str = gtk_editable_get_chars (GTK_EDITABLE (widget), start, end); gdk_clipboard_set_text (gtk_widget_get_clipboard (widget), str); g_free (str); g_dbus_method_invocation_return_value (invocation, NULL); } else if (g_strcmp0 (method_name, "CutText") == 0) { int start, end; gboolean ret = FALSE; g_variant_get (parameters, "(ii)", &start, &end); if (gtk_editable_get_editable (GTK_EDITABLE (widget))) { char *str; str = gtk_editable_get_chars (GTK_EDITABLE (widget), start, end); gdk_clipboard_set_text (gtk_widget_get_clipboard (widget), str); g_free (str); gtk_editable_delete_text (GTK_EDITABLE (widget), start, end); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "DeleteText") == 0) { int start, end; gboolean ret = FALSE; g_variant_get (parameters, "(ii)", &start, &end); if (gtk_editable_get_editable (GTK_EDITABLE (widget))) { gtk_editable_delete_text (GTK_EDITABLE (widget), start, end); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "PasteText") == 0) { int position; gboolean ret = FALSE; g_variant_get (parameters, "(i)", &position); if (gtk_editable_get_editable (GTK_EDITABLE (widget))) { PasteData *data; data = g_new (PasteData, 1); data->widget = widget; data->position = position; gdk_clipboard_read_text_async (gtk_widget_get_clipboard (widget), NULL, text_received, data); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } } static const GDBusInterfaceVTable entry_vtable = { entry_handle_method, NULL, }; /* }}} */ /* {{{ GtkTextView */ static void text_view_received (GObject *source, GAsyncResult *result, gpointer data) { GdkClipboard *clipboard = GDK_CLIPBOARD (source); PasteData *pdata = data; GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (pdata->widget)); GtkTextIter iter; char *text; text = gdk_clipboard_read_text_finish (clipboard, result, NULL); if (text) { gtk_text_buffer_get_iter_at_offset (buffer, &iter, pdata->position); gtk_text_buffer_insert (buffer, &iter, text, -1); } g_free (text); g_free (pdata); } static void text_view_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, "SetTextContents") == 0) { GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); char *text; gboolean ret = FALSE; g_variant_get (parameters, "(&s)", &text); if (gtk_text_view_get_editable (GTK_TEXT_VIEW (widget))) { gtk_text_buffer_set_text (buffer, text, -1); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "InsertText") == 0) { GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); GtkTextIter iter; int position; char *text; int len; gboolean ret = FALSE; g_variant_get (parameters, "(i&si)", &position, &text, &len); if (gtk_text_view_get_editable (GTK_TEXT_VIEW (widget))) { gtk_text_buffer_get_iter_at_offset (buffer, &iter, position); gtk_text_buffer_insert (buffer, &iter, text, len); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "CopyText") == 0) { GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); GtkTextIter start_iter, end_iter; int start, end; char *str; g_variant_get (parameters, "(ii)", &start, &end); gtk_text_buffer_get_iter_at_offset (buffer, &start_iter, start); gtk_text_buffer_get_iter_at_offset (buffer, &end_iter, end); str = gtk_text_buffer_get_text (buffer, &start_iter, &end_iter, FALSE); gdk_clipboard_set_text (gtk_widget_get_clipboard (widget), str); g_free (str); g_dbus_method_invocation_return_value (invocation, NULL); } else if (g_strcmp0 (method_name, "CutText") == 0) { GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); GtkTextIter start_iter, end_iter; int start, end; gboolean ret = FALSE; g_variant_get (parameters, "(ii)", &start, &end); if (gtk_text_view_get_editable (GTK_TEXT_VIEW (widget))) { char *str; gtk_text_buffer_get_iter_at_offset (buffer, &start_iter, start); gtk_text_buffer_get_iter_at_offset (buffer, &end_iter, end); str = gtk_text_buffer_get_text (buffer, &start_iter, &end_iter, FALSE); gdk_clipboard_set_text (gtk_widget_get_clipboard (widget), str); g_free (str); gtk_text_buffer_delete (buffer, &start_iter, &end_iter); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "DeleteText") == 0) { GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)); GtkTextIter start_iter, end_iter; int start, end; gboolean ret = FALSE; g_variant_get (parameters, "(ii)", &start, &end); if (gtk_text_view_get_editable (GTK_TEXT_VIEW (widget))) { gtk_text_buffer_get_iter_at_offset (buffer, &start_iter, start); gtk_text_buffer_get_iter_at_offset (buffer, &end_iter, end); gtk_text_buffer_delete (buffer, &start_iter, &end_iter); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } else if (g_strcmp0 (method_name, "PasteText") == 0) { int position; gboolean ret = FALSE; g_variant_get (parameters, "(i)", &position); if (gtk_text_view_get_editable (GTK_TEXT_VIEW (widget))) { PasteData *data; data = g_new (PasteData, 1); data->widget = widget; data->position = position; gdk_clipboard_read_text_async (gtk_widget_get_clipboard (widget), NULL, text_view_received, data); ret = TRUE; } g_dbus_method_invocation_return_value (invocation, g_variant_new ("(b)", ret)); } } static const GDBusInterfaceVTable text_view_vtable = { text_view_handle_method, NULL, }; /* }}} */ const GDBusInterfaceVTable * gtk_atspi_get_editable_text_vtable (GtkAccessible *accessible) { if (GTK_IS_ENTRY (accessible) || GTK_IS_SEARCH_ENTRY (accessible) || GTK_IS_PASSWORD_ENTRY (accessible) || GTK_IS_SPIN_BUTTON (accessible)) return &entry_vtable; else if (GTK_IS_TEXT_VIEW (accessible)) return &text_view_vtable; return NULL; } /* vim:set foldmethod=marker expandtab: */