123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- /*
- ==============================================================================
- 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.
- ==============================================================================
- */
- DrawableImage::DrawableImage()
- : opacity (1.0f),
- overlayColour (0x00000000)
- {
- bounds.topRight = RelativePoint (Point<float> (1.0f, 0.0f));
- bounds.bottomLeft = RelativePoint (Point<float> (0.0f, 1.0f));
- }
- DrawableImage::DrawableImage (const DrawableImage& other)
- : Drawable (other),
- image (other.image),
- opacity (other.opacity),
- overlayColour (other.overlayColour),
- bounds (other.bounds)
- {
- setBounds (other.getBounds());
- }
- DrawableImage::~DrawableImage()
- {
- }
- //==============================================================================
- void DrawableImage::setImage (const Image& imageToUse)
- {
- image = imageToUse;
- setBounds (imageToUse.getBounds());
- bounds.topLeft = RelativePoint (Point<float> (0.0f, 0.0f));
- bounds.topRight = RelativePoint (Point<float> ((float) image.getWidth(), 0.0f));
- bounds.bottomLeft = RelativePoint (Point<float> (0.0f, (float) image.getHeight()));
- recalculateCoordinates (nullptr);
- repaint();
- }
- void DrawableImage::setOpacity (const float newOpacity)
- {
- opacity = newOpacity;
- }
- void DrawableImage::setOverlayColour (Colour newOverlayColour)
- {
- overlayColour = newOverlayColour;
- }
- void DrawableImage::setBoundingBox (const RelativeParallelogram& newBounds)
- {
- if (bounds != newBounds)
- {
- bounds = newBounds;
- if (bounds.isDynamic())
- {
- Drawable::Positioner<DrawableImage>* const p = new Drawable::Positioner<DrawableImage> (*this);
- setPositioner (p);
- p->apply();
- }
- else
- {
- setPositioner (nullptr);
- recalculateCoordinates (nullptr);
- }
- }
- }
- //==============================================================================
- bool DrawableImage::registerCoordinates (RelativeCoordinatePositionerBase& pos)
- {
- bool ok = pos.addPoint (bounds.topLeft);
- ok = pos.addPoint (bounds.topRight) && ok;
- return pos.addPoint (bounds.bottomLeft) && ok;
- }
- void DrawableImage::recalculateCoordinates (Expression::Scope* scope)
- {
- if (image.isValid())
- {
- Point<float> resolved[3];
- bounds.resolveThreePoints (resolved, scope);
- const Point<float> tr (resolved[0] + (resolved[1] - resolved[0]) / (float) image.getWidth());
- const Point<float> bl (resolved[0] + (resolved[2] - resolved[0]) / (float) image.getHeight());
- AffineTransform t (AffineTransform::fromTargetPoints (resolved[0].x, resolved[0].y,
- tr.x, tr.y,
- bl.x, bl.y));
- if (t.isSingularity())
- t = AffineTransform();
- setTransform (t);
- }
- }
- //==============================================================================
- void DrawableImage::paint (Graphics& g)
- {
- if (image.isValid())
- {
- if (opacity > 0.0f && ! overlayColour.isOpaque())
- {
- g.setOpacity (opacity);
- g.drawImageAt (image, 0, 0, false);
- }
- if (! overlayColour.isTransparent())
- {
- g.setColour (overlayColour.withMultipliedAlpha (opacity));
- g.drawImageAt (image, 0, 0, true);
- }
- }
- }
- Rectangle<float> DrawableImage::getDrawableBounds() const
- {
- return image.getBounds().toFloat();
- }
- bool DrawableImage::hitTest (int x, int y)
- {
- return Drawable::hitTest (x, y) && image.isValid() && image.getPixelAt (x, y).getAlpha() >= 127;
- }
- Drawable* DrawableImage::createCopy() const
- {
- return new DrawableImage (*this);
- }
- //==============================================================================
- const Identifier DrawableImage::valueTreeType ("Image");
- const Identifier DrawableImage::ValueTreeWrapper::opacity ("opacity");
- const Identifier DrawableImage::ValueTreeWrapper::overlay ("overlay");
- const Identifier DrawableImage::ValueTreeWrapper::image ("image");
- const Identifier DrawableImage::ValueTreeWrapper::topLeft ("topLeft");
- const Identifier DrawableImage::ValueTreeWrapper::topRight ("topRight");
- const Identifier DrawableImage::ValueTreeWrapper::bottomLeft ("bottomLeft");
- //==============================================================================
- DrawableImage::ValueTreeWrapper::ValueTreeWrapper (const ValueTree& state_)
- : ValueTreeWrapperBase (state_)
- {
- jassert (state.hasType (valueTreeType));
- }
- var DrawableImage::ValueTreeWrapper::getImageIdentifier() const
- {
- return state [image];
- }
- Value DrawableImage::ValueTreeWrapper::getImageIdentifierValue (UndoManager* undoManager)
- {
- return state.getPropertyAsValue (image, undoManager);
- }
- void DrawableImage::ValueTreeWrapper::setImageIdentifier (const var& newIdentifier, UndoManager* undoManager)
- {
- state.setProperty (image, newIdentifier, undoManager);
- }
- float DrawableImage::ValueTreeWrapper::getOpacity() const
- {
- return (float) state.getProperty (opacity, 1.0);
- }
- Value DrawableImage::ValueTreeWrapper::getOpacityValue (UndoManager* undoManager)
- {
- if (! state.hasProperty (opacity))
- state.setProperty (opacity, 1.0, undoManager);
- return state.getPropertyAsValue (opacity, undoManager);
- }
- void DrawableImage::ValueTreeWrapper::setOpacity (float newOpacity, UndoManager* undoManager)
- {
- state.setProperty (opacity, newOpacity, undoManager);
- }
- Colour DrawableImage::ValueTreeWrapper::getOverlayColour() const
- {
- return Colour::fromString (state [overlay].toString());
- }
- void DrawableImage::ValueTreeWrapper::setOverlayColour (Colour newColour, UndoManager* undoManager)
- {
- if (newColour.isTransparent())
- state.removeProperty (overlay, undoManager);
- else
- state.setProperty (overlay, String::toHexString ((int) newColour.getARGB()), undoManager);
- }
- Value DrawableImage::ValueTreeWrapper::getOverlayColourValue (UndoManager* undoManager)
- {
- return state.getPropertyAsValue (overlay, undoManager);
- }
- RelativeParallelogram DrawableImage::ValueTreeWrapper::getBoundingBox() const
- {
- return RelativeParallelogram (state.getProperty (topLeft, "0, 0"),
- state.getProperty (topRight, "100, 0"),
- state.getProperty (bottomLeft, "0, 100"));
- }
- void DrawableImage::ValueTreeWrapper::setBoundingBox (const RelativeParallelogram& newBounds, UndoManager* undoManager)
- {
- state.setProperty (topLeft, newBounds.topLeft.toString(), undoManager);
- state.setProperty (topRight, newBounds.topRight.toString(), undoManager);
- state.setProperty (bottomLeft, newBounds.bottomLeft.toString(), undoManager);
- }
- //==============================================================================
- void DrawableImage::refreshFromValueTree (const ValueTree& tree, ComponentBuilder& builder)
- {
- const ValueTreeWrapper controller (tree);
- setComponentID (controller.getID());
- const float newOpacity = controller.getOpacity();
- const Colour newOverlayColour (controller.getOverlayColour());
- Image newImage;
- const var imageIdentifier (controller.getImageIdentifier());
- jassert (builder.getImageProvider() != 0 || imageIdentifier.isVoid()); // if you're using images, you need to provide something that can load and save them!
- if (builder.getImageProvider() != nullptr)
- newImage = builder.getImageProvider()->getImageForIdentifier (imageIdentifier);
- const RelativeParallelogram newBounds (controller.getBoundingBox());
- if (bounds != newBounds || newOpacity != opacity
- || overlayColour != newOverlayColour || image != newImage)
- {
- repaint();
- opacity = newOpacity;
- overlayColour = newOverlayColour;
- if (image != newImage)
- setImage (newImage);
- setBoundingBox (newBounds);
- }
- }
- ValueTree DrawableImage::createValueTree (ComponentBuilder::ImageProvider* imageProvider) const
- {
- ValueTree tree (valueTreeType);
- ValueTreeWrapper v (tree);
- v.setID (getComponentID());
- v.setOpacity (opacity, nullptr);
- v.setOverlayColour (overlayColour, nullptr);
- v.setBoundingBox (bounds, nullptr);
- if (image.isValid())
- {
- jassert (imageProvider != nullptr); // if you're using images, you need to provide something that can load and save them!
- if (imageProvider != nullptr)
- v.setImageIdentifier (imageProvider->getIdentifierForImage (image), nullptr);
- }
- return tree;
- }
|