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.

510 lines
15KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  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 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. MultiDocumentPanelWindow::MultiDocumentPanelWindow (Colour backgroundColour)
  20. : DocumentWindow (String(), backgroundColour,
  21. DocumentWindow::maximiseButton | DocumentWindow::closeButton, false)
  22. {
  23. }
  24. MultiDocumentPanelWindow::~MultiDocumentPanelWindow()
  25. {
  26. }
  27. //==============================================================================
  28. void MultiDocumentPanelWindow::maximiseButtonPressed()
  29. {
  30. if (MultiDocumentPanel* const owner = getOwner())
  31. owner->setLayoutMode (MultiDocumentPanel::MaximisedWindowsWithTabs);
  32. else
  33. jassertfalse; // these windows are only designed to be used inside a MultiDocumentPanel!
  34. }
  35. void MultiDocumentPanelWindow::closeButtonPressed()
  36. {
  37. if (MultiDocumentPanel* const owner = getOwner())
  38. owner->closeDocument (getContentComponent(), true);
  39. else
  40. jassertfalse; // these windows are only designed to be used inside a MultiDocumentPanel!
  41. }
  42. void MultiDocumentPanelWindow::activeWindowStatusChanged()
  43. {
  44. DocumentWindow::activeWindowStatusChanged();
  45. updateOrder();
  46. }
  47. void MultiDocumentPanelWindow::broughtToFront()
  48. {
  49. DocumentWindow::broughtToFront();
  50. updateOrder();
  51. }
  52. void MultiDocumentPanelWindow::updateOrder()
  53. {
  54. if (MultiDocumentPanel* const owner = getOwner())
  55. owner->updateOrder();
  56. }
  57. MultiDocumentPanel* MultiDocumentPanelWindow::getOwner() const noexcept
  58. {
  59. return findParentComponentOfClass<MultiDocumentPanel>();
  60. }
  61. //==============================================================================
  62. class MultiDocumentPanel::TabbedComponentInternal : public TabbedComponent
  63. {
  64. public:
  65. TabbedComponentInternal()
  66. : TabbedComponent (TabbedButtonBar::TabsAtTop)
  67. {
  68. }
  69. void currentTabChanged (int, const String&)
  70. {
  71. if (MultiDocumentPanel* const owner = findParentComponentOfClass<MultiDocumentPanel>())
  72. owner->updateOrder();
  73. }
  74. };
  75. //==============================================================================
  76. MultiDocumentPanel::MultiDocumentPanel()
  77. : mode (MaximisedWindowsWithTabs),
  78. backgroundColour (Colours::lightblue),
  79. maximumNumDocuments (0),
  80. numDocsBeforeTabsUsed (0)
  81. {
  82. setOpaque (true);
  83. }
  84. MultiDocumentPanel::~MultiDocumentPanel()
  85. {
  86. closeAllDocuments (false);
  87. }
  88. //==============================================================================
  89. namespace MultiDocHelpers
  90. {
  91. static bool shouldDeleteComp (Component* const c)
  92. {
  93. return c->getProperties() ["mdiDocumentDelete_"];
  94. }
  95. }
  96. bool MultiDocumentPanel::closeAllDocuments (const bool checkItsOkToCloseFirst)
  97. {
  98. while (components.size() > 0)
  99. if (! closeDocument (components.getLast(), checkItsOkToCloseFirst))
  100. return false;
  101. return true;
  102. }
  103. MultiDocumentPanelWindow* MultiDocumentPanel::createNewDocumentWindow()
  104. {
  105. return new MultiDocumentPanelWindow (backgroundColour);
  106. }
  107. void MultiDocumentPanel::addWindow (Component* component)
  108. {
  109. MultiDocumentPanelWindow* const dw = createNewDocumentWindow();
  110. dw->setResizable (true, false);
  111. dw->setContentNonOwned (component, true);
  112. dw->setName (component->getName());
  113. const var bkg (component->getProperties() ["mdiDocumentBkg_"]);
  114. dw->setBackgroundColour (bkg.isVoid() ? backgroundColour : Colour ((uint32) static_cast<int> (bkg)));
  115. int x = 4;
  116. if (Component* const topComp = getChildComponent (getNumChildComponents() - 1))
  117. if (topComp->getX() == x && topComp->getY() == x)
  118. x += 16;
  119. dw->setTopLeftPosition (x, x);
  120. const var pos (component->getProperties() ["mdiDocumentPos_"]);
  121. if (pos.toString().isNotEmpty())
  122. dw->restoreWindowStateFromString (pos.toString());
  123. addAndMakeVisible (dw);
  124. dw->toFront (true);
  125. }
  126. bool MultiDocumentPanel::addDocument (Component* const component,
  127. Colour docColour,
  128. const bool deleteWhenRemoved)
  129. {
  130. // If you try passing a full DocumentWindow or ResizableWindow in here, you'll end up
  131. // with a frame-within-a-frame! Just pass in the bare content component.
  132. jassert (dynamic_cast<ResizableWindow*> (component) == nullptr);
  133. if (component == nullptr || (maximumNumDocuments > 0 && components.size() >= maximumNumDocuments))
  134. return false;
  135. components.add (component);
  136. component->getProperties().set ("mdiDocumentDelete_", deleteWhenRemoved);
  137. component->getProperties().set ("mdiDocumentBkg_", (int) docColour.getARGB());
  138. component->addComponentListener (this);
  139. if (mode == FloatingWindows)
  140. {
  141. if (isFullscreenWhenOneDocument())
  142. {
  143. if (components.size() == 1)
  144. {
  145. addAndMakeVisible (component);
  146. }
  147. else
  148. {
  149. if (components.size() == 2)
  150. addWindow (components.getFirst());
  151. addWindow (component);
  152. }
  153. }
  154. else
  155. {
  156. addWindow (component);
  157. }
  158. }
  159. else
  160. {
  161. if (tabComponent == nullptr && components.size() > numDocsBeforeTabsUsed)
  162. {
  163. addAndMakeVisible (tabComponent = new TabbedComponentInternal());
  164. Array <Component*> temp (components);
  165. for (int i = 0; i < temp.size(); ++i)
  166. tabComponent->addTab (temp[i]->getName(), docColour, temp[i], false);
  167. resized();
  168. }
  169. else
  170. {
  171. if (tabComponent != nullptr)
  172. tabComponent->addTab (component->getName(), docColour, component, false);
  173. else
  174. addAndMakeVisible (component);
  175. }
  176. setActiveDocument (component);
  177. }
  178. resized();
  179. activeDocumentChanged();
  180. return true;
  181. }
  182. bool MultiDocumentPanel::closeDocument (Component* component,
  183. const bool checkItsOkToCloseFirst)
  184. {
  185. if (components.contains (component))
  186. {
  187. if (checkItsOkToCloseFirst && ! tryToCloseDocument (component))
  188. return false;
  189. component->removeComponentListener (this);
  190. const bool shouldDelete = MultiDocHelpers::shouldDeleteComp (component);
  191. component->getProperties().remove ("mdiDocumentDelete_");
  192. component->getProperties().remove ("mdiDocumentBkg_");
  193. if (mode == FloatingWindows)
  194. {
  195. for (int i = getNumChildComponents(); --i >= 0;)
  196. {
  197. if (MultiDocumentPanelWindow* const dw = dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)))
  198. {
  199. if (dw->getContentComponent() == component)
  200. {
  201. ScopedPointer<MultiDocumentPanelWindow> (dw)->clearContentComponent();
  202. break;
  203. }
  204. }
  205. }
  206. if (shouldDelete)
  207. delete component;
  208. components.removeFirstMatchingValue (component);
  209. if (isFullscreenWhenOneDocument() && components.size() == 1)
  210. {
  211. for (int i = getNumChildComponents(); --i >= 0;)
  212. {
  213. ScopedPointer<MultiDocumentPanelWindow> dw (dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)));
  214. if (dw != nullptr)
  215. dw->clearContentComponent();
  216. }
  217. addAndMakeVisible (components.getFirst());
  218. }
  219. }
  220. else
  221. {
  222. jassert (components.indexOf (component) >= 0);
  223. if (tabComponent != nullptr)
  224. {
  225. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  226. if (tabComponent->getTabContentComponent (i) == component)
  227. tabComponent->removeTab (i);
  228. }
  229. else
  230. {
  231. removeChildComponent (component);
  232. }
  233. if (shouldDelete)
  234. delete component;
  235. if (tabComponent != nullptr && tabComponent->getNumTabs() <= numDocsBeforeTabsUsed)
  236. tabComponent = nullptr;
  237. components.removeFirstMatchingValue (component);
  238. if (components.size() > 0 && tabComponent == nullptr)
  239. addAndMakeVisible (components.getFirst());
  240. }
  241. resized();
  242. // This ensures that the active tab is painted properly when a tab is closed!
  243. if (Component* activeComponent = getActiveDocument())
  244. setActiveDocument (activeComponent);
  245. activeDocumentChanged();
  246. }
  247. else
  248. {
  249. jassertfalse;
  250. }
  251. return true;
  252. }
  253. int MultiDocumentPanel::getNumDocuments() const noexcept
  254. {
  255. return components.size();
  256. }
  257. Component* MultiDocumentPanel::getDocument (const int index) const noexcept
  258. {
  259. return components [index];
  260. }
  261. Component* MultiDocumentPanel::getActiveDocument() const noexcept
  262. {
  263. if (mode == FloatingWindows)
  264. {
  265. for (int i = getNumChildComponents(); --i >= 0;)
  266. if (MultiDocumentPanelWindow* const dw = dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)))
  267. if (dw->isActiveWindow())
  268. return dw->getContentComponent();
  269. }
  270. return components.getLast();
  271. }
  272. void MultiDocumentPanel::setActiveDocument (Component* component)
  273. {
  274. jassert (component != nullptr);
  275. if (mode == FloatingWindows)
  276. {
  277. component = getContainerComp (component);
  278. if (component != nullptr)
  279. component->toFront (true);
  280. }
  281. else if (tabComponent != nullptr)
  282. {
  283. jassert (components.indexOf (component) >= 0);
  284. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  285. {
  286. if (tabComponent->getTabContentComponent (i) == component)
  287. {
  288. tabComponent->setCurrentTabIndex (i);
  289. break;
  290. }
  291. }
  292. }
  293. else
  294. {
  295. component->grabKeyboardFocus();
  296. }
  297. }
  298. void MultiDocumentPanel::activeDocumentChanged()
  299. {
  300. }
  301. void MultiDocumentPanel::setMaximumNumDocuments (const int newNumber)
  302. {
  303. maximumNumDocuments = newNumber;
  304. }
  305. void MultiDocumentPanel::useFullscreenWhenOneDocument (const bool shouldUseTabs)
  306. {
  307. numDocsBeforeTabsUsed = shouldUseTabs ? 1 : 0;
  308. }
  309. bool MultiDocumentPanel::isFullscreenWhenOneDocument() const noexcept
  310. {
  311. return numDocsBeforeTabsUsed != 0;
  312. }
  313. //==============================================================================
  314. void MultiDocumentPanel::setLayoutMode (const LayoutMode newLayoutMode)
  315. {
  316. if (mode != newLayoutMode)
  317. {
  318. mode = newLayoutMode;
  319. if (mode == FloatingWindows)
  320. {
  321. tabComponent = nullptr;
  322. }
  323. else
  324. {
  325. for (int i = getNumChildComponents(); --i >= 0;)
  326. {
  327. ScopedPointer<MultiDocumentPanelWindow> dw (dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)));
  328. if (dw != nullptr)
  329. {
  330. dw->getContentComponent()->getProperties().set ("mdiDocumentPos_", dw->getWindowStateAsString());
  331. dw->clearContentComponent();
  332. }
  333. }
  334. }
  335. resized();
  336. const Array <Component*> tempComps (components);
  337. components.clear();
  338. for (int i = 0; i < tempComps.size(); ++i)
  339. {
  340. Component* const c = tempComps.getUnchecked(i);
  341. addDocument (c,
  342. Colour ((uint32) static_cast<int> (c->getProperties().getWithDefault ("mdiDocumentBkg_", (int) Colours::white.getARGB()))),
  343. MultiDocHelpers::shouldDeleteComp (c));
  344. }
  345. }
  346. }
  347. void MultiDocumentPanel::setBackgroundColour (Colour newBackgroundColour)
  348. {
  349. if (backgroundColour != newBackgroundColour)
  350. {
  351. backgroundColour = newBackgroundColour;
  352. setOpaque (newBackgroundColour.isOpaque());
  353. repaint();
  354. }
  355. }
  356. //==============================================================================
  357. void MultiDocumentPanel::paint (Graphics& g)
  358. {
  359. g.fillAll (backgroundColour);
  360. }
  361. void MultiDocumentPanel::resized()
  362. {
  363. if (mode == MaximisedWindowsWithTabs || components.size() == numDocsBeforeTabsUsed)
  364. {
  365. for (int i = getNumChildComponents(); --i >= 0;)
  366. getChildComponent (i)->setBounds (getLocalBounds());
  367. }
  368. setWantsKeyboardFocus (components.size() == 0);
  369. }
  370. Component* MultiDocumentPanel::getContainerComp (Component* c) const
  371. {
  372. if (mode == FloatingWindows)
  373. {
  374. for (int i = 0; i < getNumChildComponents(); ++i)
  375. if (MultiDocumentPanelWindow* const dw = dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)))
  376. if (dw->getContentComponent() == c)
  377. return dw;
  378. }
  379. return c;
  380. }
  381. void MultiDocumentPanel::componentNameChanged (Component&)
  382. {
  383. if (mode == FloatingWindows)
  384. {
  385. for (int i = 0; i < getNumChildComponents(); ++i)
  386. if (MultiDocumentPanelWindow* const dw = dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)))
  387. dw->setName (dw->getContentComponent()->getName());
  388. }
  389. else if (tabComponent != nullptr)
  390. {
  391. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  392. tabComponent->setTabName (i, tabComponent->getTabContentComponent (i)->getName());
  393. }
  394. }
  395. void MultiDocumentPanel::updateOrder()
  396. {
  397. const Array <Component*> oldList (components);
  398. if (mode == FloatingWindows)
  399. {
  400. components.clear();
  401. for (int i = 0; i < getNumChildComponents(); ++i)
  402. if (MultiDocumentPanelWindow* const dw = dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)))
  403. components.add (dw->getContentComponent());
  404. }
  405. else
  406. {
  407. if (tabComponent != nullptr)
  408. {
  409. if (Component* const current = tabComponent->getCurrentContentComponent())
  410. {
  411. components.removeFirstMatchingValue (current);
  412. components.add (current);
  413. }
  414. }
  415. }
  416. if (components != oldList)
  417. activeDocumentChanged();
  418. }