/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2017 - ROLI Ltd. 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 5 End-User License Agreement and JUCE 5 Privacy Policy (both updated and effective as of the 27th April 2017). End User License Agreement: www.juce.com/juce-5-licence Privacy Policy: www.juce.com/juce-5-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 { ComponentMovementWatcher::ComponentMovementWatcher (Component* const comp) : component (comp), lastPeerID (0), reentrant (false), wasShowing (comp->isShowing()) { jassert (component != nullptr); // can't use this with a null pointer.. component->addComponentListener (this); registerWithParentComps(); } ComponentMovementWatcher::~ComponentMovementWatcher() { if (component != nullptr) component->removeComponentListener (this); unregister(); } //============================================================================== void ComponentMovementWatcher::componentParentHierarchyChanged (Component&) { if (component != nullptr && ! reentrant) { const ScopedValueSetter setter (reentrant, true); ComponentPeer* const peer = component->getPeer(); const uint32 peerID = peer != nullptr ? peer->getUniqueID() : 0; if (peerID != lastPeerID) { componentPeerChanged(); if (component == nullptr) return; lastPeerID = peerID; } unregister(); registerWithParentComps(); componentMovedOrResized (*component, true, true); if (component != nullptr) componentVisibilityChanged (*component); } } void ComponentMovementWatcher::componentMovedOrResized (Component&, bool wasMoved, bool wasResized) { if (component != nullptr) { if (wasMoved) { Point newPos; Component* const top = component->getTopLevelComponent(); if (top != component) newPos = top->getLocalPoint (component, Point()); else newPos = top->getPosition(); wasMoved = lastBounds.getPosition() != newPos; lastBounds.setPosition (newPos); } wasResized = (lastBounds.getWidth() != component->getWidth() || lastBounds.getHeight() != component->getHeight()); lastBounds.setSize (component->getWidth(), component->getHeight()); if (wasMoved || wasResized) componentMovedOrResized (wasMoved, wasResized); } } void ComponentMovementWatcher::componentBeingDeleted (Component& comp) { registeredParentComps.removeFirstMatchingValue (&comp); if (component == &comp) unregister(); } void ComponentMovementWatcher::componentVisibilityChanged (Component&) { if (component != nullptr) { const bool isShowingNow = component->isShowing(); if (wasShowing != isShowingNow) { wasShowing = isShowingNow; componentVisibilityChanged(); } } } void ComponentMovementWatcher::registerWithParentComps() { for (Component* p = component->getParentComponent(); p != nullptr; p = p->getParentComponent()) { p->addComponentListener (this); registeredParentComps.add (p); } } void ComponentMovementWatcher::unregister() { for (int i = registeredParentComps.size(); --i >= 0;) registeredParentComps.getUnchecked(i)->removeComponentListener (this); registeredParentComps.clear(); } } // namespace juce