/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2022 - Raw Material Software Limited JUCE is an open source library subject to commercial or open-source licensing. By using JUCE, you agree to the terms of both the JUCE 7 End-User License Agreement and JUCE Privacy Policy. End User License Agreement: www.juce.com/juce-7-licence Privacy Policy: www.juce.com/juce-privacy-policy Or: You may also use this code under the terms of the GPL v3 (see www.gnu.org/licenses). JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE DISCLAIMED. ============================================================================== */ namespace juce { //============================================================================== namespace KeyboardFocusTraverserHelpers { static bool isKeyboardFocusable (const Component* comp, const Component* container) { return comp->getWantsKeyboardFocus() && container->isParentOf (comp); } static Component* traverse (Component* current, Component* container, FocusHelpers::NavigationDirection direction) { if (auto* comp = FocusHelpers::navigateFocus (current, container, direction, &Component::isKeyboardFocusContainer)) { if (isKeyboardFocusable (comp, container)) return comp; return traverse (comp, container, direction); } return nullptr; } } Component* KeyboardFocusTraverser::getNextComponent (Component* current) { return KeyboardFocusTraverserHelpers::traverse (current, current->findKeyboardFocusContainer(), FocusHelpers::NavigationDirection::forwards); } Component* KeyboardFocusTraverser::getPreviousComponent (Component* current) { return KeyboardFocusTraverserHelpers::traverse (current, current->findKeyboardFocusContainer(), FocusHelpers::NavigationDirection::backwards); } Component* KeyboardFocusTraverser::getDefaultComponent (Component* parentComponent) { for (auto* comp : getAllComponents (parentComponent)) if (KeyboardFocusTraverserHelpers::isKeyboardFocusable (comp, parentComponent)) return comp; return nullptr; } std::vector KeyboardFocusTraverser::getAllComponents (Component* parentComponent) { std::vector components; FocusHelpers::findAllComponents (parentComponent, components, &Component::isKeyboardFocusContainer); auto removePredicate = [parentComponent] (const Component* comp) { return ! KeyboardFocusTraverserHelpers::isKeyboardFocusable (comp, parentComponent); }; components.erase (std::remove_if (std::begin (components), std::end (components), std::move (removePredicate)), std::end (components)); return components; } //============================================================================== //============================================================================== #if JUCE_UNIT_TESTS struct KeyboardFocusTraverserTests : public UnitTest { KeyboardFocusTraverserTests() : UnitTest ("KeyboardFocusTraverser", UnitTestCategories::gui) {} void runTest() override { ScopedJuceInitialiser_GUI libraryInitialiser; const MessageManagerLock mml; beginTest ("No child wants keyboard focus"); { TestComponent parent; expect (traverser.getDefaultComponent (&parent) == nullptr); expect (traverser.getAllComponents (&parent).empty()); } beginTest ("Single child wants keyboard focus"); { TestComponent parent; parent.children[5].setWantsKeyboardFocus (true); auto* defaultComponent = traverser.getDefaultComponent (&parent); expect (defaultComponent == &parent.children[5]); expect (defaultComponent->getWantsKeyboardFocus()); expect (traverser.getNextComponent (defaultComponent) == nullptr); expect (traverser.getPreviousComponent (defaultComponent) == nullptr); expect (traverser.getAllComponents (&parent).size() == 1); } beginTest ("Multiple children want keyboard focus"); { TestComponent parent; Component* focusChildren[] { &parent.children[1], &parent.children[9], &parent.children[3], &parent.children[5], &parent.children[8], &parent.children[0] }; for (auto* focusChild : focusChildren) focusChild->setWantsKeyboardFocus (true); auto allComponents = traverser.getAllComponents (&parent); for (auto* focusChild : focusChildren) expect (std::find (allComponents.cbegin(), allComponents.cend(), focusChild) != allComponents.cend()); auto* componentToTest = traverser.getDefaultComponent (&parent); for (;;) { expect (componentToTest->getWantsKeyboardFocus()); expect (std::find (std::begin (focusChildren), std::end (focusChildren), componentToTest) != std::end (focusChildren)); componentToTest = traverser.getNextComponent (componentToTest); if (componentToTest == nullptr) break; } int focusOrder = 1; for (auto* focusChild : focusChildren) focusChild->setExplicitFocusOrder (focusOrder++); componentToTest = traverser.getDefaultComponent (&parent); for (auto* focusChild : focusChildren) { expect (componentToTest == focusChild); expect (componentToTest->getWantsKeyboardFocus()); componentToTest = traverser.getNextComponent (componentToTest); } } beginTest ("Single nested child wants keyboard focus"); { TestComponent parent; Component grandparent; grandparent.addAndMakeVisible (parent); auto& focusChild = parent.children[5]; focusChild.setWantsKeyboardFocus (true); expect (traverser.getDefaultComponent (&grandparent) == &focusChild); expect (traverser.getDefaultComponent (&parent) == &focusChild); expect (traverser.getNextComponent (&focusChild) == nullptr); expect (traverser.getPreviousComponent (&focusChild) == nullptr); expect (traverser.getAllComponents (&parent).size() == 1); } beginTest ("Multiple nested children want keyboard focus"); { TestComponent parent; Component grandparent; grandparent.addAndMakeVisible (parent); Component* focusChildren[] { &parent.children[1], &parent.children[4], &parent.children[5] }; for (auto* focusChild : focusChildren) focusChild->setWantsKeyboardFocus (true); auto allComponents = traverser.getAllComponents (&parent); expect (std::equal (allComponents.cbegin(), allComponents.cend(), focusChildren, [] (const Component* c1, const Component* c2) { return c1 == c2; })); const auto front = *focusChildren; const auto back = *std::prev (std::end (focusChildren)); expect (traverser.getDefaultComponent (&grandparent) == front); expect (traverser.getDefaultComponent (&parent) == front); expect (traverser.getNextComponent (front) == *std::next (std::begin (focusChildren))); expect (traverser.getPreviousComponent (back) == *std::prev (std::end (focusChildren), 2)); std::array otherParents; for (auto& p : otherParents) { grandparent.addAndMakeVisible (p); p.setWantsKeyboardFocus (true); } expect (traverser.getDefaultComponent (&grandparent) == front); expect (traverser.getDefaultComponent (&parent) == front); expect (traverser.getNextComponent (back) == &otherParents.front()); expect (traverser.getNextComponent (&otherParents.back()) == nullptr); expect (traverser.getAllComponents (&grandparent).size() == numElementsInArray (focusChildren) + otherParents.size()); expect (traverser.getAllComponents (&parent).size() == (size_t) numElementsInArray (focusChildren)); for (auto* focusChild : focusChildren) focusChild->setWantsKeyboardFocus (false); expect (traverser.getDefaultComponent (&grandparent) == &otherParents.front()); expect (traverser.getDefaultComponent (&parent) == nullptr); expect (traverser.getAllComponents (&grandparent).size() == otherParents.size()); expect (traverser.getAllComponents (&parent).empty()); } } private: struct TestComponent : public Component { TestComponent() { for (auto& child : children) addAndMakeVisible (child); } std::array children; }; KeyboardFocusTraverser traverser; }; static KeyboardFocusTraverserTests keyboardFocusTraverserTests; #endif } // namespace juce