|
- /*
- ==============================================================================
-
- 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 FocusHelpers
- {
- static int getOrder (const Component* c)
- {
- auto order = c->getExplicitFocusOrder();
- return order > 0 ? order : std::numeric_limits<int>::max();
- }
-
- template <typename FocusContainerFn>
- static void findAllComponents (Component* parent,
- std::vector<Component*>& components,
- FocusContainerFn isFocusContainer)
- {
- if (parent == nullptr || parent->getNumChildComponents() == 0)
- return;
-
- std::vector<Component*> localComponents;
-
- for (auto* c : parent->getChildren())
- if (c->isVisible() && c->isEnabled())
- localComponents.push_back (c);
-
- const auto compareComponents = [&] (const Component* a, const Component* b)
- {
- const auto getComponentOrderAttributes = [] (const Component* c)
- {
- return std::make_tuple (getOrder (c),
- c->isAlwaysOnTop() ? 0 : 1,
- c->getY(),
- c->getX());
- };
-
- return getComponentOrderAttributes (a) < getComponentOrderAttributes (b);
- };
-
- // This will sort so that they are ordered in terms of explicit focus,
- // always on top, left-to-right, and then top-to-bottom.
- std::stable_sort (localComponents.begin(), localComponents.end(), compareComponents);
-
- for (auto* c : localComponents)
- {
- components.push_back (c);
-
- if (! (c->*isFocusContainer)())
- findAllComponents (c, components, isFocusContainer);
- }
- }
-
- enum class NavigationDirection { forwards, backwards };
-
- template <typename FocusContainerFn>
- static Component* navigateFocus (Component* current,
- Component* focusContainer,
- NavigationDirection direction,
- FocusContainerFn isFocusContainer)
- {
- if (focusContainer != nullptr)
- {
- std::vector<Component*> components;
- findAllComponents (focusContainer, components, isFocusContainer);
-
- const auto iter = std::find (components.cbegin(), components.cend(), current);
-
- if (iter == components.cend())
- return nullptr;
-
- switch (direction)
- {
- case NavigationDirection::forwards:
- if (iter != std::prev (components.cend()))
- return *std::next (iter);
-
- break;
-
- case NavigationDirection::backwards:
- if (iter != components.cbegin())
- return *std::prev (iter);
-
- break;
- }
- }
-
- return nullptr;
- }
- }
-
- //==============================================================================
- Component* FocusTraverser::getNextComponent (Component* current)
- {
- jassert (current != nullptr);
-
- return FocusHelpers::navigateFocus (current,
- current->findFocusContainer(),
- FocusHelpers::NavigationDirection::forwards,
- &Component::isFocusContainer);
- }
-
- Component* FocusTraverser::getPreviousComponent (Component* current)
- {
- jassert (current != nullptr);
-
- return FocusHelpers::navigateFocus (current,
- current->findFocusContainer(),
- FocusHelpers::NavigationDirection::backwards,
- &Component::isFocusContainer);
- }
-
- Component* FocusTraverser::getDefaultComponent (Component* parentComponent)
- {
- if (parentComponent != nullptr)
- {
- std::vector<Component*> components;
- FocusHelpers::findAllComponents (parentComponent,
- components,
- &Component::isFocusContainer);
-
- if (! components.empty())
- return components.front();
- }
-
- return nullptr;
- }
-
- std::vector<Component*> FocusTraverser::getAllComponents (Component* parentComponent)
- {
- std::vector<Component*> components;
- FocusHelpers::findAllComponents (parentComponent,
- components,
- &Component::isFocusContainer);
-
- return components;
- }
-
- //==============================================================================
- //==============================================================================
- #if JUCE_UNIT_TESTS
-
- struct FocusTraverserTests : public UnitTest
- {
- FocusTraverserTests()
- : UnitTest ("FocusTraverser", UnitTestCategories::gui)
- {}
-
- void runTest() override
- {
- ScopedJuceInitialiser_GUI libraryInitialiser;
- const MessageManagerLock mml;
-
- beginTest ("Basic traversal");
- {
- TestComponent parent;
-
- expect (traverser.getDefaultComponent (&parent) == &parent.children.front());
-
- for (auto iter = parent.children.begin(); iter != parent.children.end(); ++iter)
- expect (traverser.getNextComponent (&(*iter)) == (iter == std::prev (parent.children.cend()) ? nullptr
- : &(*std::next (iter))));
-
- for (auto iter = parent.children.rbegin(); iter != parent.children.rend(); ++iter)
- expect (traverser.getPreviousComponent (&(*iter)) == (iter == std::prev (parent.children.rend()) ? nullptr
- : &(*std::next (iter))));
- auto allComponents = traverser.getAllComponents (&parent);
-
- expect (std::equal (allComponents.cbegin(), allComponents.cend(), parent.children.cbegin(),
- [] (const Component* c1, const Component& c2) { return c1 == &c2; }));
- }
-
- beginTest ("Disabled components are ignored");
- {
- checkIgnored ([] (Component& c) { c.setEnabled (false); });
- }
-
- beginTest ("Invisible components are ignored");
- {
- checkIgnored ([] (Component& c) { c.setVisible (false); });
- }
-
- beginTest ("Explicit focus order comes before unspecified");
- {
- TestComponent parent;
-
- auto& explicitFocusComponent = parent.children[2];
-
- explicitFocusComponent.setExplicitFocusOrder (1);
- expect (traverser.getDefaultComponent (&parent) == &explicitFocusComponent);
-
- expect (traverser.getAllComponents (&parent).front() == &explicitFocusComponent);
- }
-
- beginTest ("Explicit focus order comparison");
- {
- checkComponentProperties ([this] (Component& child) { child.setExplicitFocusOrder (getRandom().nextInt ({ 1, 100 })); },
- [] (const Component& c1, const Component& c2) { return c1.getExplicitFocusOrder()
- <= c2.getExplicitFocusOrder(); });
- }
-
- beginTest ("Left to right");
- {
- checkComponentProperties ([this] (Component& child) { child.setTopLeftPosition (getRandom().nextInt ({ 0, 100 }), 0); },
- [] (const Component& c1, const Component& c2) { return c1.getX() <= c2.getX(); });
- }
-
- beginTest ("Top to bottom");
- {
- checkComponentProperties ([this] (Component& child) { child.setTopLeftPosition (0, getRandom().nextInt ({ 0, 100 })); },
- [] (const Component& c1, const Component& c2) { return c1.getY() <= c2.getY(); });
- }
-
- beginTest ("Focus containers have their own focus");
- {
- Component root;
-
- TestComponent container;
- container.setFocusContainerType (Component::FocusContainerType::focusContainer);
-
- root.addAndMakeVisible (container);
-
- expect (traverser.getDefaultComponent (&root) == &container);
- expect (traverser.getNextComponent (&container) == nullptr);
- expect (traverser.getPreviousComponent (&container) == nullptr);
-
- expect (traverser.getDefaultComponent (&container) == &container.children.front());
-
- for (auto iter = container.children.begin(); iter != container.children.end(); ++iter)
- expect (traverser.getNextComponent (&(*iter)) == (iter == std::prev (container.children.cend()) ? nullptr
- : &(*std::next (iter))));
-
- for (auto iter = container.children.rbegin(); iter != container.children.rend(); ++iter)
- expect (traverser.getPreviousComponent (&(*iter)) == (iter == std::prev (container.children.rend()) ? nullptr
- : &(*std::next (iter))));
-
- expect (traverser.getAllComponents (&root).size() == 1);
-
- auto allContainerComponents = traverser.getAllComponents (&container);
-
- expect (std::equal (allContainerComponents.cbegin(), allContainerComponents.cend(), container.children.cbegin(),
- [] (const Component* c1, const Component& c2) { return c1 == &c2; }));
- }
-
- beginTest ("Non-focus containers pass-through focus");
- {
- Component root;
-
- TestComponent container;
- container.setFocusContainerType (Component::FocusContainerType::none);
-
- root.addAndMakeVisible (container);
-
- expect (traverser.getDefaultComponent (&root) == &container);
- expect (traverser.getNextComponent (&container) == &container.children.front());
- expect (traverser.getPreviousComponent (&container) == nullptr);
-
- expect (traverser.getDefaultComponent (&container) == &container.children.front());
-
- for (auto iter = container.children.begin(); iter != container.children.end(); ++iter)
- expect (traverser.getNextComponent (&(*iter)) == (iter == std::prev (container.children.cend()) ? nullptr
- : &(*std::next (iter))));
-
- for (auto iter = container.children.rbegin(); iter != container.children.rend(); ++iter)
- expect (traverser.getPreviousComponent (&(*iter)) == (iter == std::prev (container.children.rend()) ? &container
- : &(*std::next (iter))));
-
- expect (traverser.getAllComponents (&root).size() == container.children.size() + 1);
- }
- }
-
- private:
- struct TestComponent : public Component
- {
- TestComponent()
- {
- for (auto& child : children)
- addAndMakeVisible (child);
- }
-
- std::array<Component, 10> children;
- };
-
- void checkComponentProperties (std::function<void (Component&)>&& childFn,
- std::function<bool (const Component&, const Component&)>&& testProperty)
- {
- TestComponent parent;
-
- for (auto& child : parent.children)
- childFn (child);
-
- auto* comp = traverser.getDefaultComponent (&parent);
-
- for (const auto& child : parent.children)
- if (&child != comp)
- expect (testProperty (*comp, child));
-
- for (;;)
- {
- auto* next = traverser.getNextComponent (comp);
-
- if (next == nullptr)
- break;
-
- expect (testProperty (*comp, *next));
- comp = next;
- }
- }
-
- void checkIgnored (const std::function<void(Component&)>& makeIgnored)
- {
- TestComponent parent;
-
- auto iter = parent.children.begin();
-
- makeIgnored (*iter);
- expect (traverser.getDefaultComponent (&parent) == std::addressof (*std::next (iter)));
-
- iter += 5;
- makeIgnored (*iter);
- expect (traverser.getNextComponent (std::addressof (*std::prev (iter))) == std::addressof (*std::next (iter)));
- expect (traverser.getPreviousComponent (std::addressof (*std::next (iter))) == std::addressof (*std::prev (iter)));
-
- auto allComponents = traverser.getAllComponents (&parent);
-
- expect (std::find (allComponents.cbegin(), allComponents.cend(), &parent.children.front()) == allComponents.cend());
- expect (std::find (allComponents.cbegin(), allComponents.cend(), std::addressof (*iter)) == allComponents.cend());
- }
-
- FocusTraverser traverser;
- };
-
- static FocusTraverserTests focusTraverserTests;
-
- #endif
-
- } // namespace juce
|