/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2013 - Raw Material Software 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_JUCEHEADER__ #define __JUCER_PAINTELEMENTGROUP_JUCEHEADER__ #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 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 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 (subElements.getUnchecked(i))) if (pg->containsElement (element)) return true; return false; } //============================================================================== void setInitialBounds (int /*parentWidth*/, int /*parentHeight*/) { } Rectangle getCurrentBounds (const Rectangle& parentArea) const { Rectangle 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& b, const Rectangle& parentArea, const bool undoable) { Rectangle newBounds (b); newBounds.setSize (jmax (1, newBounds.getWidth()), jmax (1, newBounds.getHeight())); const Rectangle 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 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& parentArea) { for (int i = 0; i < subElements.size(); ++i) subElements.getUnchecked(i)->draw (g, layout, parentArea); } void getEditableProperties (Array& 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 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_JUCEHEADER__