blob: 709d98494477b8458bd816d9cc825eb184d8b4a2 [file] [log] [blame]
/*
* Copyright (C) 2021 Igalia S.L.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "AccessibilityObjectAtspi.h"
#if USE(ATSPI)
#include "AccessibilityAtspi.h"
#include "AccessibilityObject.h" // NOLINT: check-webkit-style has problems with files that do not have primary header
namespace WebCore {
GDBusInterfaceVTable AccessibilityObjectAtspi::s_selectionFunctions = {
// method_call
[](GDBusConnection*, const gchar*, const gchar*, const gchar*, const gchar* methodName, GVariant* parameters, GDBusMethodInvocation* invocation, gpointer userData) {
auto atspiObject = Ref { *static_cast<AccessibilityObjectAtspi*>(userData) };
atspiObject->updateBackingStore();
if (!g_strcmp0(methodName, "GetSelectedChild")) {
int index;
g_variant_get(parameters, "(i)", &index);
auto* child = index >= 0 ? atspiObject->selectedChild(index) : nullptr;
g_dbus_method_invocation_return_value(invocation, g_variant_new("(@(so))", child ? child->reference() : AccessibilityAtspi::singleton().nullReference()));
} else if (!g_strcmp0(methodName, "SelectChild")) {
int index;
g_variant_get(parameters, "(i)", &index);
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", index >= 0 ? atspiObject->setChildSelected(index, true) : FALSE));
} else if (!g_strcmp0(methodName, "DeselectSelectedChild")) {
int index;
g_variant_get(parameters, "(i)", &index);
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", index >= 0 ? atspiObject->deselectSelectedChild(index) : FALSE));
} else if (!g_strcmp0(methodName, "IsChildSelected")) {
int index;
g_variant_get(parameters, "(i)", &index);
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", index >= 0 ? atspiObject->isChildSelected(index) : FALSE));
} else if (!g_strcmp0(methodName, "SelectAll"))
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", atspiObject->selectAll()));
else if (!g_strcmp0(methodName, "ClearSelection"))
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", atspiObject->clearSelection()));
else if (!g_strcmp0(methodName, "DeselectChild")) {
int index;
g_variant_get(parameters, "(i)", &index);
g_dbus_method_invocation_return_value(invocation, g_variant_new("(b)", index >= 0 ? atspiObject->setChildSelected(index, false) : FALSE));
}
},
// get_property
[](GDBusConnection*, const gchar*, const gchar*, const gchar*, const gchar* propertyName, GError** error, gpointer userData) -> GVariant* {
auto atspiObject = Ref { *static_cast<AccessibilityObjectAtspi*>(userData) };
atspiObject->updateBackingStore();
if (!g_strcmp0(propertyName, "NSelectedChildren"))
return g_variant_new_int32(atspiObject->selectionCount());
g_set_error(error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, "Unknown property '%s'", propertyName);
return nullptr;
},
// set_property,
nullptr,
// padding
{ nullptr }
};
unsigned AccessibilityObjectAtspi::selectionCount() const
{
if (!m_coreObject)
return 0;
AccessibilityObject::AccessibilityChildrenVector selectedItems;
m_coreObject->selectedChildren(selectedItems);
return selectedItems.size();
}
AccessibilityObjectAtspi* AccessibilityObjectAtspi::selectedChild(unsigned index) const
{
if (!m_coreObject)
return nullptr;
AccessibilityObject::AccessibilityChildrenVector selectedItems;
m_coreObject->selectedChildren(selectedItems);
if (index >= selectedItems.size())
return nullptr;
return selectedItems[index]->wrapper();
}
bool AccessibilityObjectAtspi::setChildSelected(unsigned index, bool selected) const
{
if (!m_coreObject)
return false;
const auto& children = m_coreObject->children();
if (index >= children.size())
return false;
if (!children[index]->canSetSelectedAttribute())
return false;
children[index]->setSelected(selected);
return selected ? children[index]->isSelected() : !children[index]->isSelected();
}
bool AccessibilityObjectAtspi::deselectSelectedChild(unsigned index) const
{
if (!m_coreObject)
return false;
AccessibilityObject::AccessibilityChildrenVector selectedItems;
m_coreObject->selectedChildren(selectedItems);
if (index >= selectedItems.size())
return false;
if (!selectedItems[index]->canSetSelectedAttribute())
return false;
selectedItems[index]->setSelected(false);
return !selectedItems[index]->isSelected();
}
bool AccessibilityObjectAtspi::isChildSelected(unsigned index) const
{
if (!m_coreObject)
return false;
const auto& children = m_coreObject->children();
if (index >= children.size())
return false;
return children[index]->isSelected();
}
bool AccessibilityObjectAtspi::selectAll() const
{
if (!m_coreObject)
return false;
if (!m_coreObject->isMultiSelectable() || !m_coreObject->canSetSelectedChildren())
return false;
const auto& children = m_coreObject->children();
unsigned selectableChildCount = 0;
for (const auto& child : children) {
if (child->canSetSelectedAttribute())
selectableChildCount++;
}
if (!selectableChildCount)
return false;
m_coreObject->setSelectedChildren(children);
AccessibilityObject::AccessibilityChildrenVector selectedItems;
m_coreObject->selectedChildren(selectedItems);
return selectableChildCount == selectedItems.size();
}
bool AccessibilityObjectAtspi::clearSelection() const
{
if (!m_coreObject)
return false;
if (!m_coreObject->canSetSelectedChildren())
return false;
m_coreObject->setSelectedChildren({ });
AccessibilityObject::AccessibilityChildrenVector selectedItems;
m_coreObject->selectedChildren(selectedItems);
return selectedItems.isEmpty();
}
void AccessibilityObjectAtspi::selectionChanged()
{
// selectionChanged can be called multiple times by WebCore, so ensure we don't
// emit it if the last one happened in the same run loop iteration.
auto* source = g_main_current_source();
int64_t sourceTime = source ? g_source_get_time(source) : -1;
if (sourceTime <= m_lastSelectionChangedTime)
return;
m_lastSelectionChangedTime = sourceTime;
AccessibilityAtspi::singleton().selectionChanged(*this);
}
} // namespace WebCore
#endif // USE(ATSPI)