AccessibilityTreeNode.cpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. /*
  2. * Copyright (c) 2022, Jonah Shafran <jonahshafran@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Tuple.h>
  7. #include <LibWeb/DOM/AccessibilityTreeNode.h>
  8. #include <LibWeb/DOM/Document.h>
  9. #include <LibWeb/DOM/Element.h>
  10. #include <LibWeb/DOM/Node.h>
  11. #include <LibWeb/DOM/Text.h>
  12. namespace Web::DOM {
  13. JS_DEFINE_ALLOCATOR(AccessibilityTreeNode);
  14. JS::NonnullGCPtr<AccessibilityTreeNode> AccessibilityTreeNode::create(Document* document, DOM::Node const* value)
  15. {
  16. return document->heap().allocate<AccessibilityTreeNode>(document->realm(), value);
  17. }
  18. AccessibilityTreeNode::AccessibilityTreeNode(JS::GCPtr<DOM::Node const> value)
  19. : m_value(value)
  20. {
  21. m_children = {};
  22. }
  23. void AccessibilityTreeNode::serialize_tree_as_json(JsonObjectSerializer<StringBuilder>& object, Document const& document) const
  24. {
  25. if (value()->is_document()) {
  26. VERIFY_NOT_REACHED();
  27. } else if (value()->is_element()) {
  28. auto const* element = static_cast<DOM::Element const*>(value().ptr());
  29. if (element->include_in_accessibility_tree()) {
  30. MUST(object.add("type"sv, "element"sv));
  31. auto role = element->role_or_default();
  32. bool has_role = role.has_value() && !ARIA::is_abstract_role(*role);
  33. auto name = MUST(element->accessible_name(document));
  34. MUST(object.add("name"sv, name));
  35. auto description = MUST(element->accessible_description(document));
  36. MUST(object.add("description"sv, description));
  37. MUST(object.add("id"sv, element->unique_id()));
  38. if (has_role)
  39. MUST(object.add("role"sv, ARIA::role_name(*role)));
  40. else
  41. MUST(object.add("role"sv, ""sv));
  42. } else {
  43. VERIFY_NOT_REACHED();
  44. }
  45. } else if (value()->is_text()) {
  46. MUST(object.add("type"sv, "text"sv));
  47. auto const* text_node = static_cast<DOM::Text const*>(value().ptr());
  48. MUST(object.add("text"sv, text_node->data()));
  49. }
  50. if (value()->has_child_nodes()) {
  51. auto node_children = MUST(object.add_array("children"sv));
  52. for (auto& child : children()) {
  53. if (child->value()->is_uninteresting_whitespace_node())
  54. continue;
  55. JsonObjectSerializer<StringBuilder> child_object = MUST(node_children.add_object());
  56. child->serialize_tree_as_json(child_object, document);
  57. MUST(child_object.finish());
  58. }
  59. MUST(node_children.finish());
  60. }
  61. }
  62. void AccessibilityTreeNode::visit_edges(Visitor& visitor)
  63. {
  64. Base::visit_edges(visitor);
  65. visitor.visit(m_value);
  66. for (auto const& child : m_children)
  67. visitor.visit(child);
  68. }
  69. }