The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

429 lines
14KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2022 - Raw Material Software Limited
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 7 End-User License
  8. Agreement and JUCE Privacy Policy.
  9. End User License Agreement: www.juce.com/juce-7-licence
  10. Privacy Policy: www.juce.com/juce-privacy-policy
  11. Or: You may also use this code under the terms of the GPL v3 (see
  12. www.gnu.org/licenses).
  13. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  14. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  15. DISCLAIMED.
  16. ==============================================================================
  17. */
  18. #include "../../Application/jucer_Headers.h"
  19. #include "jucer_PaintElementImage.h"
  20. PaintElementImage::PaintElementImage (PaintRoutine* pr)
  21. : PaintElement (pr, "Image"),
  22. opacity (1.0),
  23. mode (stretched)
  24. {
  25. }
  26. PaintElementImage::~PaintElementImage() {}
  27. const Drawable* PaintElementImage::getDrawable()
  28. {
  29. if (JucerDocument* const document = getDocument())
  30. return document->getResources().getDrawable (resourceName);
  31. return nullptr;
  32. }
  33. void PaintElementImage::draw (Graphics& g, const ComponentLayout* layout, const Rectangle<int>& parentArea)
  34. {
  35. const Rectangle<int> r (position.getRectangle (parentArea, layout));
  36. if (const Drawable* const image = getDrawable())
  37. {
  38. image->drawWithin (g, r.toFloat(),
  39. mode == stretched ? RectanglePlacement::stretchToFit
  40. : (mode == proportionalReducingOnly ? (RectanglePlacement::centred | RectanglePlacement::onlyReduceInSize)
  41. : RectanglePlacement::centred),
  42. (float) opacity);
  43. }
  44. else
  45. {
  46. g.setColour (Colours::grey.withAlpha (0.5f));
  47. g.fillRect (r);
  48. g.setColour (Colours::black);
  49. g.drawText ("(image missing)",
  50. r.getX(), r.getY(), r.getWidth(), r.getHeight(),
  51. Justification::centred, true);
  52. }
  53. }
  54. //==============================================================================
  55. void PaintElementImage::getEditableProperties (Array <PropertyComponent*>& props, bool multipleSelected)
  56. {
  57. PaintElement::getEditableProperties (props, multipleSelected);
  58. props.add (new ImageElementResourceProperty (this));
  59. props.add (new StretchModeProperty (this));
  60. props.add (new OpacityProperty (this));
  61. props.add (new ResetSizeProperty (this));
  62. }
  63. void PaintElementImage::fillInGeneratedCode (GeneratedCode& code, String& paintMethodCode)
  64. {
  65. if (opacity > 0)
  66. {
  67. String x, y, w, h, r;
  68. positionToCode (position, getDocument()->getComponentLayout(), x, y, w, h);
  69. r << "{\n"
  70. << " int x = " << x << ", y = " << y << ", width = " << w << ", height = " << h << ";\n"
  71. << " //[UserPaintCustomArguments] Customize the painting arguments here..\n"
  72. << customPaintCode
  73. << " //[/UserPaintCustomArguments]\n";
  74. if (dynamic_cast<const DrawableImage*> (getDrawable()))
  75. {
  76. const String imageVariable ("cachedImage_" + resourceName.replace ("::", "_") + "_" + String (code.getUniqueSuffix()));
  77. code.addImageResourceLoader (imageVariable, resourceName);
  78. if (opacity >= 254.0 / 255.0)
  79. r << " g.setColour (juce::Colours::black);\n";
  80. else
  81. r << " g.setColour (juce::Colours::black.withAlpha (" << CodeHelpers::floatLiteral (opacity, 3) << "));\n";
  82. if (mode == stretched)
  83. {
  84. r << " g.drawImage (" << imageVariable << ",\n"
  85. << " x, y, width, height,\n"
  86. << " 0, 0, " << imageVariable << ".getWidth(), " << imageVariable << ".getHeight());\n";
  87. }
  88. else
  89. {
  90. r << " g.drawImageWithin (" << imageVariable << ",\n"
  91. << " x, y, width, height,\n"
  92. << " ";
  93. if (mode == proportionalReducingOnly)
  94. r << "juce::RectanglePlacement::centred | juce::RectanglePlacement::onlyReduceInSize";
  95. else
  96. r << "juce::RectanglePlacement::centred";
  97. r << ",\n"
  98. << " false);\n";
  99. }
  100. }
  101. else
  102. {
  103. if (resourceName.isNotEmpty())
  104. {
  105. const String imageVariable ("drawable" + String (code.getUniqueSuffix()));
  106. code.privateMemberDeclarations
  107. << "std::unique_ptr<juce::Drawable> " << imageVariable << ";\n";
  108. code.constructorCode
  109. << imageVariable << " = juce::Drawable::createFromImageData ("
  110. << resourceName << ", " << resourceName << "Size);\n";
  111. code.destructorCode
  112. << imageVariable << " = nullptr;\n";
  113. if (opacity >= 254.0 / 255.0)
  114. r << " g.setColour (juce::Colours::black);\n";
  115. else
  116. r << " g.setColour (juce::Colours::black.withAlpha (" << CodeHelpers::floatLiteral (opacity, 3) << "));\n";
  117. r << " jassert (" << imageVariable << " != nullptr);\n"
  118. << " if (" << imageVariable << " != nullptr)\n"
  119. << " " << imageVariable << "->drawWithin (g, juce::Rectangle<int> (x, y, width, height).toFloat(),\n"
  120. << " " << String::repeatedString (" ", imageVariable.length() + 18)
  121. << (mode == stretched ? "juce::RectanglePlacement::stretchToFit"
  122. : (mode == proportionalReducingOnly ? "juce::RectanglePlacement::centred | juce::RectanglePlacement::onlyReduceInSize"
  123. : "juce::RectanglePlacement::centred"))
  124. << ", " << CodeHelpers::floatLiteral (opacity, 3) << ");\n";
  125. }
  126. }
  127. r << "}\n\n";
  128. paintMethodCode += r;
  129. }
  130. }
  131. void PaintElementImage::applyCustomPaintSnippets (StringArray& snippets)
  132. {
  133. customPaintCode.clear();
  134. if (! snippets.isEmpty() && opacity > 0)
  135. {
  136. customPaintCode = snippets[0];
  137. snippets.remove (0);
  138. }
  139. }
  140. //==============================================================================
  141. PaintElementImage::SetResourceAction::SetResourceAction (PaintElementImage* const element, const String& newResource_)
  142. : PaintElementUndoableAction <PaintElementImage> (element),
  143. newResource (newResource_)
  144. {
  145. oldResource = element->getResource();
  146. }
  147. bool PaintElementImage::SetResourceAction::perform()
  148. {
  149. showCorrectTab();
  150. getElement()->setResource (newResource, false);
  151. return true;
  152. }
  153. bool PaintElementImage::SetResourceAction::undo()
  154. {
  155. showCorrectTab();
  156. getElement()->setResource (oldResource, false);
  157. return true;
  158. }
  159. void PaintElementImage::setResource (const String& newName, const bool undoable)
  160. {
  161. if (resourceName != newName)
  162. {
  163. if (undoable)
  164. {
  165. perform (new SetResourceAction (this, newName),
  166. "Change image resource");
  167. }
  168. else
  169. {
  170. resourceName = newName;
  171. changed();
  172. }
  173. }
  174. repaint();
  175. }
  176. String PaintElementImage::getResource() const
  177. {
  178. return resourceName;
  179. }
  180. //==============================================================================
  181. PaintElementImage::SetOpacityAction::SetOpacityAction (PaintElementImage* const element, const double newOpacity_)
  182. : PaintElementUndoableAction <PaintElementImage> (element),
  183. newOpacity (newOpacity_)
  184. {
  185. oldOpacity = element->getOpacity();
  186. }
  187. bool PaintElementImage::SetOpacityAction::perform()
  188. {
  189. showCorrectTab();
  190. getElement()->setOpacity (newOpacity, false);
  191. return true;
  192. }
  193. bool PaintElementImage::SetOpacityAction::undo()
  194. {
  195. showCorrectTab();
  196. getElement()->setOpacity (oldOpacity, false);
  197. return true;
  198. }
  199. void PaintElementImage::setOpacity (double newOpacity, const bool undoable)
  200. {
  201. newOpacity = jlimit (0.0, 1.0, newOpacity);
  202. if (! approximatelyEqual (opacity, newOpacity))
  203. {
  204. if (undoable)
  205. {
  206. perform (new SetOpacityAction (this, newOpacity),
  207. "Change image opacity");
  208. }
  209. else
  210. {
  211. opacity = newOpacity;
  212. changed();
  213. }
  214. }
  215. }
  216. double PaintElementImage::getOpacity() const noexcept { return opacity; }
  217. //==============================================================================
  218. const char* PaintElementImage::getTagName() noexcept { return "IMAGE"; }
  219. void PaintElementImage::resetToImageSize()
  220. {
  221. if (const Drawable* const image = getDrawable())
  222. {
  223. if (PaintRoutineEditor* ed = dynamic_cast<PaintRoutineEditor*> (getParentComponent()))
  224. {
  225. const Rectangle<int> parentArea (ed->getComponentArea());
  226. Rectangle<int> r (getCurrentBounds (parentArea));
  227. Rectangle<float> b (image->getDrawableBounds());
  228. r.setSize ((int) (b.getWidth() + 0.999f),
  229. (int) (b.getHeight() + 0.999f));
  230. setCurrentBounds (r, parentArea, true);
  231. }
  232. }
  233. }
  234. //==============================================================================
  235. PaintElementImage::SetStretchModeAction::SetStretchModeAction (PaintElementImage* const element, const StretchMode newValue_)
  236. : PaintElementUndoableAction <PaintElementImage> (element),
  237. newValue (newValue_)
  238. {
  239. oldValue = element->getStretchMode();
  240. }
  241. bool PaintElementImage::SetStretchModeAction::perform()
  242. {
  243. showCorrectTab();
  244. getElement()->setStretchMode (newValue, false);
  245. return true;
  246. }
  247. bool PaintElementImage::SetStretchModeAction::undo()
  248. {
  249. showCorrectTab();
  250. getElement()->setStretchMode (oldValue, false);
  251. return true;
  252. }
  253. PaintElementImage::StretchMode PaintElementImage::getStretchMode() const noexcept { return mode; }
  254. void PaintElementImage::setStretchMode (const StretchMode newMode, const bool undoable)
  255. {
  256. if (mode != newMode)
  257. {
  258. if (undoable)
  259. {
  260. perform (new SetStretchModeAction (this, newMode),
  261. "Change image mode");
  262. }
  263. else
  264. {
  265. mode = newMode;
  266. changed();
  267. }
  268. }
  269. }
  270. //==============================================================================
  271. XmlElement* PaintElementImage::createXml() const
  272. {
  273. XmlElement* e = new XmlElement (getTagName());
  274. position.applyToXml (*e);
  275. e->setAttribute ("resource", resourceName);
  276. e->setAttribute ("opacity", opacity);
  277. e->setAttribute ("mode", (int) mode);
  278. return e;
  279. }
  280. bool PaintElementImage::loadFromXml (const XmlElement& xml)
  281. {
  282. if (xml.hasTagName (getTagName()))
  283. {
  284. position.restoreFromXml (xml, position);
  285. resourceName = xml.getStringAttribute ("resource", String());
  286. opacity = xml.getDoubleAttribute ("opacity", 1.0);
  287. mode = (StretchMode) xml.getIntAttribute ("mode", (int) stretched);
  288. repaint();
  289. return true;
  290. }
  291. jassertfalse;
  292. return false;
  293. }
  294. //==============================================================================
  295. PaintElementImage::ImageElementResourceProperty::ImageElementResourceProperty (PaintElementImage* const e)
  296. : ImageResourceProperty <PaintElementImage> (e, "image source")
  297. {
  298. }
  299. void PaintElementImage::ImageElementResourceProperty::setResource (const String& newName)
  300. {
  301. if (element != nullptr)
  302. element->setResource (newName, true);
  303. }
  304. String PaintElementImage::ImageElementResourceProperty::getResource() const
  305. {
  306. if (element != nullptr)
  307. return element->getResource();
  308. return {};
  309. }
  310. //==============================================================================
  311. PaintElementImage::OpacityProperty::OpacityProperty (PaintElementImage* const e)
  312. : SliderPropertyComponent ("opacity", 0.0, 1.0, 0.001),
  313. listener (e)
  314. {
  315. listener.setPropertyToRefresh (*this);
  316. }
  317. void PaintElementImage::OpacityProperty::setValue (double newValue)
  318. {
  319. listener.owner->getDocument()->getUndoManager().undoCurrentTransactionOnly();
  320. listener.owner->setOpacity (newValue, true);
  321. }
  322. double PaintElementImage::OpacityProperty::getValue() const
  323. {
  324. return listener.owner->getOpacity();
  325. }
  326. PaintElementImage::StretchModeProperty::StretchModeProperty (PaintElementImage* const e)
  327. : ChoicePropertyComponent ("stretch mode"),
  328. listener (e)
  329. {
  330. listener.setPropertyToRefresh (*this);
  331. choices.add ("Stretched to fit");
  332. choices.add ("Maintain aspect ratio");
  333. choices.add ("Maintain aspect ratio, only reduce in size");
  334. }
  335. void PaintElementImage::StretchModeProperty::setIndex (int newIndex)
  336. {
  337. listener.owner->setStretchMode ((StretchMode) newIndex, true);
  338. }
  339. int PaintElementImage::StretchModeProperty::getIndex() const
  340. {
  341. return (int) listener.owner->getStretchMode();
  342. }
  343. PaintElementImage::ResetSizeProperty::ResetSizeProperty (PaintElementImage* const e)
  344. : ButtonPropertyComponent ("reset", false),
  345. element (e)
  346. {
  347. }
  348. void PaintElementImage::ResetSizeProperty::buttonClicked()
  349. {
  350. element->resetToImageSize();
  351. }
  352. String PaintElementImage::ResetSizeProperty::getButtonText() const { return "reset to image size"; }