mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-15 00:24:19 +00:00
239 lines
7.8 KiB
C++
239 lines
7.8 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE library.
|
|
Copyright (c) 2015 - ROLI Ltd.
|
|
|
|
Permission is granted to use this software under the terms of either:
|
|
a) the GPL v2 (or any later version)
|
|
b) the Affero GPL v3
|
|
|
|
Details of these licenses can be found at: www.gnu.org/licenses
|
|
|
|
JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
------------------------------------------------------------------------------
|
|
|
|
To release a closed-source product which uses JUCE, commercial licenses are
|
|
available: visit www.juce.com for more information.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
#ifndef JUCER_PAINTELEMENTGROUP_H_INCLUDED
|
|
#define JUCER_PAINTELEMENTGROUP_H_INCLUDED
|
|
|
|
#include "jucer_PaintElement.h"
|
|
#include "../jucer_ObjectTypes.h"
|
|
|
|
|
|
//==============================================================================
|
|
class PaintElementGroup : public PaintElement
|
|
{
|
|
public:
|
|
PaintElementGroup (PaintRoutine* pr)
|
|
: PaintElement (pr, "Group")
|
|
{
|
|
}
|
|
|
|
void ungroup (const bool undoable)
|
|
{
|
|
getOwner()->getSelectedElements().deselectAll();
|
|
getOwner()->getSelectedPoints().deselectAll();
|
|
|
|
const int index = getOwner()->indexOfElement (this);
|
|
|
|
for (int i = 0; i < subElements.size(); ++i)
|
|
{
|
|
ScopedPointer<XmlElement> xml (subElements.getUnchecked(i)->createXml());
|
|
|
|
PaintElement* newOne = getOwner()->addElementFromXml (*xml, index, undoable);
|
|
getOwner()->getSelectedElements().addToSelection (newOne);
|
|
}
|
|
|
|
getOwner()->removeElement (this, undoable);
|
|
}
|
|
|
|
static void groupSelected (PaintRoutine* const routine)
|
|
{
|
|
if (routine->getSelectedElements().getNumSelected() > 1)
|
|
{
|
|
PaintElementGroup* newGroup = new PaintElementGroup (routine);
|
|
|
|
int frontIndex = -1;
|
|
|
|
for (int i = 0; i < routine->getNumElements(); ++i)
|
|
{
|
|
if (routine->getSelectedElements().isSelected (routine->getElement (i)))
|
|
{
|
|
ScopedPointer<XmlElement> xml (routine->getElement(i)->createXml());
|
|
|
|
if (PaintElement* newOne = ObjectTypes::createElementForXml (xml, routine))
|
|
newGroup->subElements.add (newOne);
|
|
|
|
if (i > frontIndex)
|
|
frontIndex = i;
|
|
}
|
|
}
|
|
|
|
routine->deleteSelected();
|
|
|
|
PaintElement* const g = routine->addNewElement (newGroup, frontIndex, true);
|
|
routine->getSelectedElements().selectOnly (g);
|
|
}
|
|
}
|
|
|
|
int getNumElements() const noexcept { return subElements.size(); }
|
|
|
|
PaintElement* getElement (const int index) const noexcept { return subElements [index]; }
|
|
|
|
int indexOfElement (const PaintElement* element) const noexcept { return subElements.indexOf (element); }
|
|
|
|
bool containsElement (const PaintElement* element) const
|
|
{
|
|
if (subElements.contains (element))
|
|
return true;
|
|
|
|
for (int i = subElements.size(); --i >= 0;)
|
|
if (PaintElementGroup* pg = dynamic_cast<PaintElementGroup*> (subElements.getUnchecked(i)))
|
|
if (pg->containsElement (element))
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
//==============================================================================
|
|
void setInitialBounds (int /*parentWidth*/, int /*parentHeight*/)
|
|
{
|
|
}
|
|
|
|
Rectangle<int> getCurrentBounds (const Rectangle<int>& parentArea) const
|
|
{
|
|
Rectangle<int> r;
|
|
|
|
if (subElements.size() > 0)
|
|
{
|
|
r = subElements.getUnchecked(0)->getCurrentBounds (parentArea);
|
|
|
|
for (int i = 1; i < subElements.size(); ++i)
|
|
r = r.getUnion (subElements.getUnchecked(i)->getCurrentBounds (parentArea));
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
void setCurrentBounds (const Rectangle<int>& b, const Rectangle<int>& parentArea, const bool undoable)
|
|
{
|
|
Rectangle<int> newBounds (b);
|
|
newBounds.setSize (jmax (1, newBounds.getWidth()),
|
|
jmax (1, newBounds.getHeight()));
|
|
|
|
const Rectangle<int> current (getCurrentBounds (parentArea));
|
|
|
|
if (newBounds != current)
|
|
{
|
|
const int dx = newBounds.getX() - current.getX();
|
|
const int dy = newBounds.getY() - current.getY();
|
|
|
|
const double scaleStartX = current.getX();
|
|
const double scaleStartY = current.getY();
|
|
const double scaleX = newBounds.getWidth() / (double) current.getWidth();
|
|
const double scaleY = newBounds.getHeight() / (double) current.getHeight();
|
|
|
|
for (int i = 0; i < subElements.size(); ++i)
|
|
{
|
|
PaintElement* const e = subElements.getUnchecked(i);
|
|
|
|
Rectangle<int> pos (e->getCurrentBounds (parentArea));
|
|
|
|
const int newX = roundToInt ((pos.getX() - scaleStartX) * scaleX + scaleStartX + dx);
|
|
const int newY = roundToInt ((pos.getY() - scaleStartY) * scaleY + scaleStartY + dy);
|
|
|
|
pos.setBounds (newX, newY,
|
|
roundToInt ((pos.getRight() - scaleStartX) * scaleX + scaleStartX + dx) - newX,
|
|
roundToInt ((pos.getBottom() - scaleStartY) * scaleY + scaleStartY + dy) - newY);
|
|
|
|
e->setCurrentBounds (pos, parentArea, undoable);
|
|
}
|
|
}
|
|
}
|
|
|
|
//==============================================================================
|
|
void draw (Graphics& g, const ComponentLayout* layout, const Rectangle<int>& parentArea)
|
|
{
|
|
for (int i = 0; i < subElements.size(); ++i)
|
|
subElements.getUnchecked(i)->draw (g, layout, parentArea);
|
|
}
|
|
|
|
void getEditableProperties (Array<PropertyComponent*>& props)
|
|
{
|
|
props.add (new UngroupProperty (this));
|
|
}
|
|
|
|
void fillInGeneratedCode (GeneratedCode& code, String& paintMethodCode)
|
|
{
|
|
for (int i = 0; i < subElements.size(); ++i)
|
|
subElements.getUnchecked(i)->fillInGeneratedCode (code, paintMethodCode);
|
|
}
|
|
|
|
static const char* getTagName() noexcept { return "GROUP"; }
|
|
|
|
XmlElement* createXml() const
|
|
{
|
|
XmlElement* e = new XmlElement (getTagName());
|
|
|
|
for (int i = 0; i < subElements.size(); ++i)
|
|
{
|
|
XmlElement* const sub = subElements.getUnchecked(i)->createXml();
|
|
e->addChildElement (sub);
|
|
}
|
|
|
|
return e;
|
|
}
|
|
|
|
bool loadFromXml (const XmlElement& xml)
|
|
{
|
|
if (xml.hasTagName (getTagName()))
|
|
{
|
|
forEachXmlChildElement (xml, e)
|
|
if (PaintElement* const pe = ObjectTypes::createElementForXml (e, owner))
|
|
subElements.add (pe);
|
|
|
|
return true;
|
|
}
|
|
|
|
jassertfalse;
|
|
return false;
|
|
}
|
|
|
|
private:
|
|
OwnedArray <PaintElement> subElements;
|
|
|
|
class UngroupProperty : public ButtonPropertyComponent
|
|
{
|
|
public:
|
|
UngroupProperty (PaintElementGroup* const e)
|
|
: ButtonPropertyComponent ("ungroup", false),
|
|
element (e)
|
|
{
|
|
}
|
|
|
|
void buttonClicked()
|
|
{
|
|
element->ungroup (true);
|
|
}
|
|
|
|
String getButtonText() const
|
|
{
|
|
return "Ungroup";
|
|
}
|
|
|
|
private:
|
|
PaintElementGroup* const element;
|
|
};
|
|
};
|
|
|
|
|
|
#endif // JUCER_PAINTELEMENTGROUP_H_INCLUDED
|