2018-06-17 20:34:53 +08:00
|
|
|
/*
|
|
|
|
==============================================================================
|
|
|
|
|
|
|
|
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
|
|
|
|
{
|
|
|
|
|
2019-06-23 03:41:38 +08:00
|
|
|
ResizableBorderComponent::Zone::Zone() noexcept {}
|
|
|
|
ResizableBorderComponent::Zone::Zone (int zoneFlags) noexcept : zone (zoneFlags) {}
|
2018-06-17 20:34:53 +08:00
|
|
|
ResizableBorderComponent::Zone::Zone (const ResizableBorderComponent::Zone& other) noexcept : zone (other.zone) {}
|
|
|
|
|
|
|
|
ResizableBorderComponent::Zone& ResizableBorderComponent::Zone::operator= (const ResizableBorderComponent::Zone& other) noexcept
|
|
|
|
{
|
|
|
|
zone = other.zone;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ResizableBorderComponent::Zone::operator== (const ResizableBorderComponent::Zone& other) const noexcept { return zone == other.zone; }
|
|
|
|
bool ResizableBorderComponent::Zone::operator!= (const ResizableBorderComponent::Zone& other) const noexcept { return zone != other.zone; }
|
|
|
|
|
2019-06-23 03:41:38 +08:00
|
|
|
ResizableBorderComponent::Zone ResizableBorderComponent::Zone::fromPositionOnBorder (Rectangle<int> totalSize,
|
|
|
|
BorderSize<int> border,
|
2018-06-17 20:34:53 +08:00
|
|
|
Point<int> position)
|
|
|
|
{
|
|
|
|
int z = 0;
|
|
|
|
|
|
|
|
if (totalSize.contains (position)
|
|
|
|
&& ! border.subtractedFrom (totalSize).contains (position))
|
|
|
|
{
|
2019-06-23 03:41:38 +08:00
|
|
|
auto minW = jmax (totalSize.getWidth() / 10, jmin (10, totalSize.getWidth() / 3));
|
|
|
|
|
2018-06-17 20:34:53 +08:00
|
|
|
if (position.x < jmax (border.getLeft(), minW) && border.getLeft() > 0)
|
|
|
|
z |= left;
|
|
|
|
else if (position.x >= totalSize.getWidth() - jmax (border.getRight(), minW) && border.getRight() > 0)
|
|
|
|
z |= right;
|
|
|
|
|
2019-06-23 03:41:38 +08:00
|
|
|
auto minH = jmax (totalSize.getHeight() / 10, jmin (10, totalSize.getHeight() / 3));
|
|
|
|
|
2018-06-17 20:34:53 +08:00
|
|
|
if (position.y < jmax (border.getTop(), minH) && border.getTop() > 0)
|
|
|
|
z |= top;
|
|
|
|
else if (position.y >= totalSize.getHeight() - jmax (border.getBottom(), minH) && border.getBottom() > 0)
|
|
|
|
z |= bottom;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Zone (z);
|
|
|
|
}
|
|
|
|
|
|
|
|
MouseCursor ResizableBorderComponent::Zone::getMouseCursor() const noexcept
|
|
|
|
{
|
2019-06-23 03:41:38 +08:00
|
|
|
auto mc = MouseCursor::NormalCursor;
|
2018-06-17 20:34:53 +08:00
|
|
|
|
|
|
|
switch (zone)
|
|
|
|
{
|
|
|
|
case (left | top): mc = MouseCursor::TopLeftCornerResizeCursor; break;
|
|
|
|
case top: mc = MouseCursor::TopEdgeResizeCursor; break;
|
|
|
|
case (right | top): mc = MouseCursor::TopRightCornerResizeCursor; break;
|
|
|
|
case left: mc = MouseCursor::LeftEdgeResizeCursor; break;
|
|
|
|
case right: mc = MouseCursor::RightEdgeResizeCursor; break;
|
|
|
|
case (left | bottom): mc = MouseCursor::BottomLeftCornerResizeCursor; break;
|
|
|
|
case bottom: mc = MouseCursor::BottomEdgeResizeCursor; break;
|
|
|
|
case (right | bottom): mc = MouseCursor::BottomRightCornerResizeCursor; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mc;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
ResizableBorderComponent::ResizableBorderComponent (Component* const componentToResize,
|
|
|
|
ComponentBoundsConstrainer* const constrainer_)
|
|
|
|
: component (componentToResize),
|
|
|
|
constrainer (constrainer_),
|
|
|
|
borderSize (5),
|
|
|
|
mouseZone (0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ResizableBorderComponent::~ResizableBorderComponent()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
void ResizableBorderComponent::paint (Graphics& g)
|
|
|
|
{
|
|
|
|
getLookAndFeel().drawResizableFrame (g, getWidth(), getHeight(), borderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::mouseEnter (const MouseEvent& e)
|
|
|
|
{
|
|
|
|
updateMouseZone (e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::mouseMove (const MouseEvent& e)
|
|
|
|
{
|
|
|
|
updateMouseZone (e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::mouseDown (const MouseEvent& e)
|
|
|
|
{
|
|
|
|
if (component == nullptr)
|
|
|
|
{
|
|
|
|
jassertfalse; // You've deleted the component that this resizer was supposed to be using!
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
updateMouseZone (e);
|
|
|
|
|
|
|
|
originalBounds = component->getBounds();
|
|
|
|
|
|
|
|
if (constrainer != nullptr)
|
|
|
|
constrainer->resizeStart();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::mouseDrag (const MouseEvent& e)
|
|
|
|
{
|
|
|
|
if (component == nullptr)
|
|
|
|
{
|
|
|
|
jassertfalse; // You've deleted the component that this resizer was supposed to be using!
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-23 03:41:38 +08:00
|
|
|
auto newBounds = mouseZone.resizeRectangleBy (originalBounds, e.getOffsetFromDragStart());
|
2018-06-17 20:34:53 +08:00
|
|
|
|
|
|
|
if (constrainer != nullptr)
|
|
|
|
{
|
|
|
|
constrainer->setBoundsForComponent (component, newBounds,
|
|
|
|
mouseZone.isDraggingTopEdge(),
|
|
|
|
mouseZone.isDraggingLeftEdge(),
|
|
|
|
mouseZone.isDraggingBottomEdge(),
|
|
|
|
mouseZone.isDraggingRightEdge());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-23 03:41:38 +08:00
|
|
|
if (auto* p = component->getPositioner())
|
|
|
|
p->applyNewBounds (newBounds);
|
2018-06-17 20:34:53 +08:00
|
|
|
else
|
|
|
|
component->setBounds (newBounds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::mouseUp (const MouseEvent&)
|
|
|
|
{
|
|
|
|
if (constrainer != nullptr)
|
|
|
|
constrainer->resizeEnd();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ResizableBorderComponent::hitTest (int x, int y)
|
|
|
|
{
|
|
|
|
return x < borderSize.getLeft()
|
|
|
|
|| x >= getWidth() - borderSize.getRight()
|
|
|
|
|| y < borderSize.getTop()
|
|
|
|
|| y >= getHeight() - borderSize.getBottom();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::setBorderThickness (const BorderSize<int>& newBorderSize)
|
|
|
|
{
|
|
|
|
if (borderSize != newBorderSize)
|
|
|
|
{
|
|
|
|
borderSize = newBorderSize;
|
|
|
|
repaint();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BorderSize<int> ResizableBorderComponent::getBorderThickness() const
|
|
|
|
{
|
|
|
|
return borderSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ResizableBorderComponent::updateMouseZone (const MouseEvent& e)
|
|
|
|
{
|
2019-06-23 03:41:38 +08:00
|
|
|
auto newZone = Zone::fromPositionOnBorder (getLocalBounds(), borderSize, e.getPosition());
|
2018-06-17 20:34:53 +08:00
|
|
|
|
|
|
|
if (mouseZone != newZone)
|
|
|
|
{
|
|
|
|
mouseZone = newZone;
|
|
|
|
setMouseCursor (newZone.getMouseCursor());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace juce
|