mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-10 23:44:24 +00:00
649 lines
22 KiB
C++
649 lines
22 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE framework.
|
|
Copyright (c) Raw Material Software Limited
|
|
|
|
JUCE is an open source framework subject to commercial or open source
|
|
licensing.
|
|
|
|
By downloading, installing, or using the JUCE framework, or combining the
|
|
JUCE framework with any other source code, object code, content or any other
|
|
copyrightable work, you agree to the terms of the JUCE End User Licence
|
|
Agreement, and all incorporated terms including the JUCE Privacy Policy and
|
|
the JUCE Website Terms of Service, as applicable, which will bind you. If you
|
|
do not agree to the terms of these agreements, we will not license the JUCE
|
|
framework to you, and you must discontinue the installation or download
|
|
process and cease use of the JUCE framework.
|
|
|
|
JUCE End User Licence Agreement: https://juce.com/legal/juce-8-licence/
|
|
JUCE Privacy Policy: https://juce.com/juce-privacy-policy
|
|
JUCE Website Terms of Service: https://juce.com/juce-website-terms-of-service/
|
|
|
|
Or:
|
|
|
|
You may also use this code under the terms of the AGPLv3:
|
|
https://www.gnu.org/licenses/agpl-3.0.en.html
|
|
|
|
THE JUCE FRAMEWORK IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL
|
|
WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING WARRANTY OF
|
|
MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, ARE DISCLAIMED.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
namespace juce
|
|
{
|
|
|
|
using ViewportDragPosition = AnimatedPosition<AnimatedPositionBehaviours::ContinuousWithMomentum>;
|
|
|
|
struct Viewport::DragToScrollListener final : private MouseListener,
|
|
private ViewportDragPosition::Listener
|
|
{
|
|
DragToScrollListener (Viewport& v) : viewport (v)
|
|
{
|
|
viewport.contentHolder.addMouseListener (this, true);
|
|
offsetX.addListener (this);
|
|
offsetY.addListener (this);
|
|
offsetX.behaviour.setMinimumVelocity (60);
|
|
offsetY.behaviour.setMinimumVelocity (60);
|
|
}
|
|
|
|
~DragToScrollListener() override
|
|
{
|
|
viewport.contentHolder.removeMouseListener (this);
|
|
Desktop::getInstance().removeGlobalMouseListener (this);
|
|
}
|
|
|
|
void stopOngoingAnimation()
|
|
{
|
|
offsetX.setPosition (offsetX.getPosition());
|
|
offsetY.setPosition (offsetY.getPosition());
|
|
}
|
|
|
|
void positionChanged (ViewportDragPosition&, double) override
|
|
{
|
|
viewport.setViewPosition (originalViewPos - Point<int> ((int) offsetX.getPosition(),
|
|
(int) offsetY.getPosition()));
|
|
}
|
|
|
|
void mouseDown (const MouseEvent& e) override
|
|
{
|
|
if (! isGlobalMouseListener && detail::ViewportHelpers::wouldScrollOnEvent (&viewport, e.source))
|
|
{
|
|
offsetX.setPosition (offsetX.getPosition());
|
|
offsetY.setPosition (offsetY.getPosition());
|
|
|
|
// switch to a global mouse listener so we still receive mouseUp events
|
|
// if the original event component is deleted
|
|
viewport.contentHolder.removeMouseListener (this);
|
|
Desktop::getInstance().addGlobalMouseListener (this);
|
|
|
|
isGlobalMouseListener = true;
|
|
|
|
scrollSource = e.source;
|
|
}
|
|
}
|
|
|
|
void mouseDrag (const MouseEvent& e) override
|
|
{
|
|
if (e.source == scrollSource
|
|
&& ! doesMouseEventComponentBlockViewportDrag (e.eventComponent))
|
|
{
|
|
auto totalOffset = e.getEventRelativeTo (&viewport).getOffsetFromDragStart().toFloat();
|
|
|
|
if (! isDragging && totalOffset.getDistanceFromOrigin() > 8.0f && detail::ViewportHelpers::wouldScrollOnEvent (&viewport, e.source))
|
|
{
|
|
isDragging = true;
|
|
|
|
originalViewPos = viewport.getViewPosition();
|
|
offsetX.setPosition (0.0);
|
|
offsetX.beginDrag();
|
|
offsetY.setPosition (0.0);
|
|
offsetY.beginDrag();
|
|
}
|
|
|
|
if (isDragging)
|
|
{
|
|
offsetX.drag (totalOffset.x);
|
|
offsetY.drag (totalOffset.y);
|
|
}
|
|
}
|
|
}
|
|
|
|
void mouseUp (const MouseEvent& e) override
|
|
{
|
|
if (isGlobalMouseListener && e.source == scrollSource)
|
|
endDragAndClearGlobalMouseListener();
|
|
}
|
|
|
|
void endDragAndClearGlobalMouseListener()
|
|
{
|
|
if (std::exchange (isDragging, false) == true)
|
|
{
|
|
offsetX.endDrag();
|
|
offsetY.endDrag();
|
|
}
|
|
|
|
viewport.contentHolder.addMouseListener (this, true);
|
|
Desktop::getInstance().removeGlobalMouseListener (this);
|
|
|
|
isGlobalMouseListener = false;
|
|
}
|
|
|
|
bool doesMouseEventComponentBlockViewportDrag (const Component* eventComp)
|
|
{
|
|
for (auto c = eventComp; c != nullptr && c != &viewport; c = c->getParentComponent())
|
|
if (c->getViewportIgnoreDragFlag())
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
Viewport& viewport;
|
|
ViewportDragPosition offsetX, offsetY;
|
|
Point<int> originalViewPos;
|
|
MouseInputSource scrollSource = Desktop::getInstance().getMainMouseSource();
|
|
bool isDragging = false;
|
|
bool isGlobalMouseListener = false;
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DragToScrollListener)
|
|
};
|
|
|
|
//==============================================================================
|
|
Viewport::Viewport (const String& name)
|
|
: Component (name),
|
|
dragToScrollListener (std::make_unique<DragToScrollListener> (*this))
|
|
{
|
|
// content holder is used to clip the contents so they don't overlap the scrollbars
|
|
addAndMakeVisible (contentHolder);
|
|
contentHolder.setInterceptsMouseClicks (false, true);
|
|
|
|
scrollBarThickness = getLookAndFeel().getDefaultScrollbarWidth();
|
|
|
|
setInterceptsMouseClicks (false, true);
|
|
setWantsKeyboardFocus (true);
|
|
|
|
recreateScrollbars();
|
|
}
|
|
|
|
Viewport::~Viewport()
|
|
{
|
|
deleteOrRemoveContentComp();
|
|
}
|
|
|
|
//==============================================================================
|
|
void Viewport::visibleAreaChanged (const Rectangle<int>&) {}
|
|
void Viewport::viewedComponentChanged (Component*) {}
|
|
|
|
//==============================================================================
|
|
void Viewport::deleteOrRemoveContentComp()
|
|
{
|
|
if (contentComp != nullptr)
|
|
{
|
|
contentComp->removeComponentListener (this);
|
|
|
|
if (deleteContent)
|
|
{
|
|
// This sets the content comp to a null pointer before deleting the old one, in case
|
|
// anything tries to use the old one while it's in mid-deletion..
|
|
std::unique_ptr<Component> oldCompDeleter (contentComp.get());
|
|
contentComp = nullptr;
|
|
}
|
|
else
|
|
{
|
|
contentHolder.removeChildComponent (contentComp);
|
|
contentComp = nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Viewport::setViewedComponent (Component* const newViewedComponent, const bool deleteComponentWhenNoLongerNeeded)
|
|
{
|
|
if (contentComp.get() != newViewedComponent)
|
|
{
|
|
deleteOrRemoveContentComp();
|
|
contentComp = newViewedComponent;
|
|
deleteContent = deleteComponentWhenNoLongerNeeded;
|
|
|
|
if (contentComp != nullptr)
|
|
{
|
|
contentHolder.addAndMakeVisible (contentComp);
|
|
setViewPosition (Point<int>());
|
|
contentComp->addComponentListener (this);
|
|
}
|
|
|
|
viewedComponentChanged (contentComp);
|
|
updateVisibleArea();
|
|
}
|
|
}
|
|
|
|
void Viewport::recreateScrollbars()
|
|
{
|
|
verticalScrollBar.reset();
|
|
horizontalScrollBar.reset();
|
|
|
|
verticalScrollBar .reset (createScrollBarComponent (true));
|
|
horizontalScrollBar.reset (createScrollBarComponent (false));
|
|
|
|
addChildComponent (verticalScrollBar.get());
|
|
addChildComponent (horizontalScrollBar.get());
|
|
|
|
getVerticalScrollBar().addListener (this);
|
|
getHorizontalScrollBar().addListener (this);
|
|
getVerticalScrollBar().addMouseListener (this, true);
|
|
getHorizontalScrollBar().addMouseListener (this, true);
|
|
|
|
resized();
|
|
}
|
|
|
|
int Viewport::getMaximumVisibleWidth() const { return contentHolder.getWidth(); }
|
|
int Viewport::getMaximumVisibleHeight() const { return contentHolder.getHeight(); }
|
|
|
|
bool Viewport::canScrollVertically() const noexcept { return contentComp->getY() < 0 || contentComp->getBottom() > getHeight(); }
|
|
bool Viewport::canScrollHorizontally() const noexcept { return contentComp->getX() < 0 || contentComp->getRight() > getWidth(); }
|
|
|
|
Point<int> Viewport::viewportPosToCompPos (Point<int> pos) const
|
|
{
|
|
jassert (contentComp != nullptr);
|
|
|
|
auto contentBounds = contentHolder.getLocalArea (contentComp.get(), contentComp->getLocalBounds());
|
|
|
|
Point<int> p (jmax (jmin (0, contentHolder.getWidth() - contentBounds.getWidth()), jmin (0, -(pos.x))),
|
|
jmax (jmin (0, contentHolder.getHeight() - contentBounds.getHeight()), jmin (0, -(pos.y))));
|
|
|
|
return p.transformedBy (contentComp->getTransform().inverted());
|
|
}
|
|
|
|
void Viewport::setViewPosition (const int xPixelsOffset, const int yPixelsOffset)
|
|
{
|
|
setViewPosition ({ xPixelsOffset, yPixelsOffset });
|
|
}
|
|
|
|
void Viewport::setViewPosition (Point<int> newPosition)
|
|
{
|
|
if (contentComp != nullptr)
|
|
contentComp->setTopLeftPosition (viewportPosToCompPos (newPosition));
|
|
}
|
|
|
|
void Viewport::setViewPositionProportionately (const double x, const double y)
|
|
{
|
|
if (contentComp != nullptr)
|
|
setViewPosition (jmax (0, roundToInt (x * (contentComp->getWidth() - getWidth()))),
|
|
jmax (0, roundToInt (y * (contentComp->getHeight() - getHeight()))));
|
|
}
|
|
|
|
bool Viewport::autoScroll (const int mouseX, const int mouseY, const int activeBorderThickness, const int maximumSpeed)
|
|
{
|
|
if (contentComp != nullptr)
|
|
{
|
|
int dx = 0, dy = 0;
|
|
|
|
if (getHorizontalScrollBar().isVisible() || canScrollHorizontally())
|
|
{
|
|
if (mouseX < activeBorderThickness)
|
|
dx = activeBorderThickness - mouseX;
|
|
else if (mouseX >= contentHolder.getWidth() - activeBorderThickness)
|
|
dx = (contentHolder.getWidth() - activeBorderThickness) - mouseX;
|
|
|
|
if (dx < 0)
|
|
dx = jmax (dx, -maximumSpeed, contentHolder.getWidth() - contentComp->getRight());
|
|
else
|
|
dx = jmin (dx, maximumSpeed, -contentComp->getX());
|
|
}
|
|
|
|
if (getVerticalScrollBar().isVisible() || canScrollVertically())
|
|
{
|
|
if (mouseY < activeBorderThickness)
|
|
dy = activeBorderThickness - mouseY;
|
|
else if (mouseY >= contentHolder.getHeight() - activeBorderThickness)
|
|
dy = (contentHolder.getHeight() - activeBorderThickness) - mouseY;
|
|
|
|
if (dy < 0)
|
|
dy = jmax (dy, -maximumSpeed, contentHolder.getHeight() - contentComp->getBottom());
|
|
else
|
|
dy = jmin (dy, maximumSpeed, -contentComp->getY());
|
|
}
|
|
|
|
if (dx != 0 || dy != 0)
|
|
{
|
|
contentComp->setTopLeftPosition (contentComp->getX() + dx,
|
|
contentComp->getY() + dy);
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Viewport::componentMovedOrResized (Component&, bool, bool)
|
|
{
|
|
updateVisibleArea();
|
|
}
|
|
|
|
//==============================================================================
|
|
void Viewport::setScrollOnDragMode (const ScrollOnDragMode mode)
|
|
{
|
|
scrollOnDragMode = mode;
|
|
}
|
|
|
|
bool Viewport::isCurrentlyScrollingOnDrag() const noexcept
|
|
{
|
|
return dragToScrollListener->isDragging;
|
|
}
|
|
|
|
//==============================================================================
|
|
void Viewport::lookAndFeelChanged()
|
|
{
|
|
if (! customScrollBarThickness)
|
|
{
|
|
scrollBarThickness = getLookAndFeel().getDefaultScrollbarWidth();
|
|
resized();
|
|
}
|
|
}
|
|
|
|
void Viewport::resized()
|
|
{
|
|
updateVisibleArea();
|
|
}
|
|
|
|
//==============================================================================
|
|
void Viewport::updateVisibleArea()
|
|
{
|
|
auto scrollbarWidth = getScrollBarThickness();
|
|
const bool canShowAnyBars = getWidth() > scrollbarWidth && getHeight() > scrollbarWidth;
|
|
const bool canShowHBar = showHScrollbar && canShowAnyBars;
|
|
const bool canShowVBar = showVScrollbar && canShowAnyBars;
|
|
|
|
bool hBarVisible = false, vBarVisible = false;
|
|
Rectangle<int> contentArea;
|
|
|
|
for (int i = 3; --i >= 0;)
|
|
{
|
|
hBarVisible = canShowHBar && ! getHorizontalScrollBar().autoHides();
|
|
vBarVisible = canShowVBar && ! getVerticalScrollBar().autoHides();
|
|
contentArea = getLocalBounds();
|
|
|
|
if (contentComp != nullptr && ! contentArea.contains (contentComp->getBounds()))
|
|
{
|
|
hBarVisible = canShowHBar && (hBarVisible || contentComp->getX() < 0 || contentComp->getRight() > contentArea.getWidth());
|
|
vBarVisible = canShowVBar && (vBarVisible || contentComp->getY() < 0 || contentComp->getBottom() > contentArea.getHeight());
|
|
|
|
if (vBarVisible)
|
|
contentArea.setWidth (getWidth() - scrollbarWidth);
|
|
|
|
if (hBarVisible)
|
|
contentArea.setHeight (getHeight() - scrollbarWidth);
|
|
|
|
if (! contentArea.contains (contentComp->getBounds()))
|
|
{
|
|
hBarVisible = canShowHBar && (hBarVisible || contentComp->getRight() > contentArea.getWidth());
|
|
vBarVisible = canShowVBar && (vBarVisible || contentComp->getBottom() > contentArea.getHeight());
|
|
}
|
|
}
|
|
|
|
if (vBarVisible) contentArea.setWidth (getWidth() - scrollbarWidth);
|
|
if (hBarVisible) contentArea.setHeight (getHeight() - scrollbarWidth);
|
|
|
|
if (! vScrollbarRight && vBarVisible)
|
|
contentArea.setX (scrollbarWidth);
|
|
|
|
if (! hScrollbarBottom && hBarVisible)
|
|
contentArea.setY (scrollbarWidth);
|
|
|
|
if (contentComp == nullptr)
|
|
{
|
|
contentHolder.setBounds (contentArea);
|
|
break;
|
|
}
|
|
|
|
auto oldContentBounds = contentComp->getBounds();
|
|
contentHolder.setBounds (contentArea);
|
|
|
|
// If the content has changed its size, that might affect our scrollbars, so go round again and re-calculate..
|
|
if (oldContentBounds == contentComp->getBounds())
|
|
break;
|
|
}
|
|
|
|
Rectangle<int> contentBounds;
|
|
|
|
if (auto cc = contentComp.get())
|
|
contentBounds = contentHolder.getLocalArea (cc, cc->getLocalBounds());
|
|
|
|
auto visibleOrigin = -contentBounds.getPosition();
|
|
|
|
auto& hbar = getHorizontalScrollBar();
|
|
auto& vbar = getVerticalScrollBar();
|
|
|
|
hbar.setBounds (contentArea.getX(), hScrollbarBottom ? contentArea.getHeight() : 0, contentArea.getWidth(), scrollbarWidth);
|
|
hbar.setRangeLimits (0.0, contentBounds.getWidth());
|
|
hbar.setCurrentRange (visibleOrigin.x, contentArea.getWidth());
|
|
hbar.setSingleStepSize (singleStepX);
|
|
|
|
if (canShowHBar && ! hBarVisible)
|
|
visibleOrigin.setX (0);
|
|
|
|
vbar.setBounds (vScrollbarRight ? contentArea.getWidth() : 0, contentArea.getY(), scrollbarWidth, contentArea.getHeight());
|
|
vbar.setRangeLimits (0.0, contentBounds.getHeight());
|
|
vbar.setCurrentRange (visibleOrigin.y, contentArea.getHeight());
|
|
vbar.setSingleStepSize (singleStepY);
|
|
|
|
if (canShowVBar && ! vBarVisible)
|
|
visibleOrigin.setY (0);
|
|
|
|
// Force the visibility *after* setting the ranges to avoid flicker caused by edge conditions in the numbers.
|
|
hbar.setVisible (hBarVisible);
|
|
vbar.setVisible (vBarVisible);
|
|
|
|
if (contentComp != nullptr)
|
|
{
|
|
auto newContentCompPos = viewportPosToCompPos (visibleOrigin);
|
|
|
|
if (contentComp->getBounds().getPosition() != newContentCompPos)
|
|
{
|
|
contentComp->setTopLeftPosition (newContentCompPos); // (this will re-entrantly call updateVisibleArea again)
|
|
return;
|
|
}
|
|
}
|
|
|
|
const Rectangle<int> visibleArea (visibleOrigin.x, visibleOrigin.y,
|
|
jmin (contentBounds.getWidth() - visibleOrigin.x, contentArea.getWidth()),
|
|
jmin (contentBounds.getHeight() - visibleOrigin.y, contentArea.getHeight()));
|
|
|
|
if (lastVisibleArea != visibleArea)
|
|
{
|
|
lastVisibleArea = visibleArea;
|
|
visibleAreaChanged (visibleArea);
|
|
}
|
|
|
|
hbar.handleUpdateNowIfNeeded();
|
|
vbar.handleUpdateNowIfNeeded();
|
|
}
|
|
|
|
//==============================================================================
|
|
void Viewport::setSingleStepSizes (const int stepX, const int stepY)
|
|
{
|
|
if (singleStepX != stepX || singleStepY != stepY)
|
|
{
|
|
singleStepX = stepX;
|
|
singleStepY = stepY;
|
|
updateVisibleArea();
|
|
}
|
|
}
|
|
|
|
void Viewport::setScrollBarsShown (const bool showVerticalScrollbarIfNeeded,
|
|
const bool showHorizontalScrollbarIfNeeded,
|
|
const bool allowVerticalScrollingWithoutScrollbar,
|
|
const bool allowHorizontalScrollingWithoutScrollbar)
|
|
{
|
|
allowScrollingWithoutScrollbarV = allowVerticalScrollingWithoutScrollbar;
|
|
allowScrollingWithoutScrollbarH = allowHorizontalScrollingWithoutScrollbar;
|
|
|
|
if (showVScrollbar != showVerticalScrollbarIfNeeded
|
|
|| showHScrollbar != showHorizontalScrollbarIfNeeded)
|
|
{
|
|
showVScrollbar = showVerticalScrollbarIfNeeded;
|
|
showHScrollbar = showHorizontalScrollbarIfNeeded;
|
|
updateVisibleArea();
|
|
}
|
|
}
|
|
|
|
void Viewport::setScrollBarThickness (const int thickness)
|
|
{
|
|
int newThickness;
|
|
|
|
// To stay compatible with the previous code: use the
|
|
// default thickness if thickness parameter is zero
|
|
// or negative
|
|
if (thickness <= 0)
|
|
{
|
|
customScrollBarThickness = false;
|
|
newThickness = getLookAndFeel().getDefaultScrollbarWidth();
|
|
}
|
|
else
|
|
{
|
|
customScrollBarThickness = true;
|
|
newThickness = thickness;
|
|
}
|
|
|
|
if (scrollBarThickness != newThickness)
|
|
{
|
|
scrollBarThickness = newThickness;
|
|
updateVisibleArea();
|
|
}
|
|
}
|
|
|
|
int Viewport::getScrollBarThickness() const
|
|
{
|
|
return scrollBarThickness;
|
|
}
|
|
|
|
void Viewport::scrollBarMoved (ScrollBar* scrollBarThatHasMoved, double newRangeStart)
|
|
{
|
|
auto newRangeStartInt = roundToInt (newRangeStart);
|
|
|
|
if (scrollBarThatHasMoved == horizontalScrollBar.get())
|
|
{
|
|
setViewPosition (newRangeStartInt, getViewPositionY());
|
|
}
|
|
else if (scrollBarThatHasMoved == verticalScrollBar.get())
|
|
{
|
|
setViewPosition (getViewPositionX(), newRangeStartInt);
|
|
}
|
|
}
|
|
|
|
void Viewport::mouseWheelMove (const MouseEvent& e, const MouseWheelDetails& wheel)
|
|
{
|
|
if (e.eventComponent == this)
|
|
if (! useMouseWheelMoveIfNeeded (e, wheel))
|
|
Component::mouseWheelMove (e, wheel);
|
|
}
|
|
|
|
void Viewport::mouseDown (const MouseEvent& e)
|
|
{
|
|
if (e.eventComponent == horizontalScrollBar.get() || e.eventComponent == verticalScrollBar.get())
|
|
dragToScrollListener->stopOngoingAnimation();
|
|
}
|
|
|
|
static int rescaleMouseWheelDistance (float distance, int singleStepSize) noexcept
|
|
{
|
|
if (approximatelyEqual (distance, 0.0f))
|
|
return 0;
|
|
|
|
distance *= 14.0f * (float) singleStepSize;
|
|
|
|
return roundToInt (distance < 0 ? jmin (distance, -1.0f)
|
|
: jmax (distance, 1.0f));
|
|
}
|
|
|
|
bool Viewport::useMouseWheelMoveIfNeeded (const MouseEvent& e, const MouseWheelDetails& wheel)
|
|
{
|
|
if (! (e.mods.isAltDown() || e.mods.isCtrlDown() || e.mods.isCommandDown()))
|
|
{
|
|
const bool canScrollVert = (allowScrollingWithoutScrollbarV || getVerticalScrollBar().isVisible());
|
|
const bool canScrollHorz = (allowScrollingWithoutScrollbarH || getHorizontalScrollBar().isVisible());
|
|
|
|
if (canScrollHorz || canScrollVert)
|
|
{
|
|
auto deltaX = rescaleMouseWheelDistance (wheel.deltaX, singleStepX);
|
|
auto deltaY = rescaleMouseWheelDistance (wheel.deltaY, singleStepY);
|
|
|
|
auto pos = getViewPosition();
|
|
|
|
if (deltaX != 0 && deltaY != 0 && canScrollHorz && canScrollVert)
|
|
{
|
|
pos.x -= deltaX;
|
|
pos.y -= deltaY;
|
|
}
|
|
else if (canScrollHorz && (deltaX != 0 || e.mods.isShiftDown() || ! canScrollVert))
|
|
{
|
|
pos.x -= deltaX != 0 ? deltaX : deltaY;
|
|
}
|
|
else if (canScrollVert && deltaY != 0)
|
|
{
|
|
pos.y -= deltaY;
|
|
}
|
|
|
|
if (pos != getViewPosition())
|
|
{
|
|
setViewPosition (pos);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static bool isUpDownKeyPress (const KeyPress& key)
|
|
{
|
|
return key == KeyPress::upKey
|
|
|| key == KeyPress::downKey
|
|
|| key == KeyPress::pageUpKey
|
|
|| key == KeyPress::pageDownKey
|
|
|| key == KeyPress::homeKey
|
|
|| key == KeyPress::endKey;
|
|
}
|
|
|
|
static bool isLeftRightKeyPress (const KeyPress& key)
|
|
{
|
|
return key == KeyPress::leftKey
|
|
|| key == KeyPress::rightKey;
|
|
}
|
|
|
|
bool Viewport::keyPressed (const KeyPress& key)
|
|
{
|
|
const bool isUpDownKey = isUpDownKeyPress (key);
|
|
|
|
if (getVerticalScrollBar().isVisible() && isUpDownKey)
|
|
return getVerticalScrollBar().keyPressed (key);
|
|
|
|
const bool isLeftRightKey = isLeftRightKeyPress (key);
|
|
|
|
if (getHorizontalScrollBar().isVisible() && (isUpDownKey || isLeftRightKey))
|
|
return getHorizontalScrollBar().keyPressed (key);
|
|
|
|
return false;
|
|
}
|
|
|
|
bool Viewport::respondsToKey (const KeyPress& key)
|
|
{
|
|
return isUpDownKeyPress (key) || isLeftRightKeyPress (key);
|
|
}
|
|
|
|
ScrollBar* Viewport::createScrollBarComponent (bool isVertical)
|
|
{
|
|
return new ScrollBar (isVertical);
|
|
}
|
|
|
|
void Viewport::setScrollBarPosition (bool verticalScrollbarOnRight,
|
|
bool horizontalScrollbarAtBottom)
|
|
{
|
|
vScrollbarRight = verticalScrollbarOnRight;
|
|
hScrollbarBottom = horizontalScrollbarAtBottom;
|
|
|
|
resized();
|
|
}
|
|
|
|
} // namespace juce
|