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.

615 lines
18KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2020 - 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 6 End-User License
  8. Agreement and JUCE Privacy Policy (both effective as of the 16th June 2020).
  9. End User License Agreement: www.juce.com/juce-6-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. namespace juce
  19. {
  20. MultiDocumentPanelWindow::MultiDocumentPanelWindow (Colour backgroundColour)
  21. : DocumentWindow (String(), backgroundColour,
  22. DocumentWindow::maximiseButton | DocumentWindow::closeButton, false)
  23. {
  24. }
  25. MultiDocumentPanelWindow::~MultiDocumentPanelWindow()
  26. {
  27. }
  28. //==============================================================================
  29. void MultiDocumentPanelWindow::maximiseButtonPressed()
  30. {
  31. if (auto* owner = getOwner())
  32. owner->setLayoutMode (MultiDocumentPanel::MaximisedWindowsWithTabs);
  33. else
  34. jassertfalse; // these windows are only designed to be used inside a MultiDocumentPanel!
  35. }
  36. void MultiDocumentPanelWindow::closeButtonPressed()
  37. {
  38. if (auto* owner = getOwner())
  39. owner->closeDocumentAsync (getContentComponent(), true, nullptr);
  40. else
  41. jassertfalse; // these windows are only designed to be used inside a MultiDocumentPanel!
  42. }
  43. void MultiDocumentPanelWindow::activeWindowStatusChanged()
  44. {
  45. DocumentWindow::activeWindowStatusChanged();
  46. updateOrder();
  47. }
  48. void MultiDocumentPanelWindow::broughtToFront()
  49. {
  50. DocumentWindow::broughtToFront();
  51. updateOrder();
  52. }
  53. void MultiDocumentPanelWindow::updateOrder()
  54. {
  55. if (auto* owner = getOwner())
  56. owner->updateOrder();
  57. }
  58. MultiDocumentPanel* MultiDocumentPanelWindow::getOwner() const noexcept
  59. {
  60. return findParentComponentOfClass<MultiDocumentPanel>();
  61. }
  62. //==============================================================================
  63. struct MultiDocumentPanel::TabbedComponentInternal : public TabbedComponent
  64. {
  65. TabbedComponentInternal() : TabbedComponent (TabbedButtonBar::TabsAtTop) {}
  66. void currentTabChanged (int, const String&) override
  67. {
  68. if (auto* owner = findParentComponentOfClass<MultiDocumentPanel>())
  69. owner->updateOrder();
  70. }
  71. };
  72. //==============================================================================
  73. MultiDocumentPanel::MultiDocumentPanel()
  74. {
  75. setOpaque (true);
  76. }
  77. MultiDocumentPanel::~MultiDocumentPanel() = default;
  78. //==============================================================================
  79. namespace MultiDocHelpers
  80. {
  81. static bool shouldDeleteComp (Component* const c)
  82. {
  83. return c->getProperties() ["mdiDocumentDelete_"];
  84. }
  85. }
  86. #if JUCE_MODAL_LOOPS_PERMITTED
  87. bool MultiDocumentPanel::closeAllDocuments (const bool checkItsOkToCloseFirst)
  88. {
  89. while (! components.isEmpty())
  90. if (! closeDocument (components.getLast(), checkItsOkToCloseFirst))
  91. return false;
  92. return true;
  93. }
  94. #endif
  95. void MultiDocumentPanel::closeLastDocumentRecursive (SafePointer<MultiDocumentPanel> parent,
  96. bool checkItsOkToCloseFirst,
  97. std::function<void (bool)> callback)
  98. {
  99. if (parent->components.isEmpty())
  100. {
  101. if (callback != nullptr)
  102. callback (true);
  103. return;
  104. }
  105. parent->closeDocumentAsync (parent->components.getLast(),
  106. checkItsOkToCloseFirst,
  107. [parent, checkItsOkToCloseFirst, callback] (bool closeResult)
  108. {
  109. if (parent == nullptr)
  110. return;
  111. if (! closeResult)
  112. {
  113. if (callback != nullptr)
  114. callback (false);
  115. return;
  116. }
  117. parent->closeLastDocumentRecursive (parent, checkItsOkToCloseFirst, std::move (callback));
  118. });
  119. }
  120. void MultiDocumentPanel::closeAllDocumentsAsync (bool checkItsOkToCloseFirst, std::function<void (bool)> callback)
  121. {
  122. closeLastDocumentRecursive (this, checkItsOkToCloseFirst, std::move (callback));
  123. }
  124. #if JUCE_MODAL_LOOPS_PERMITTED
  125. bool MultiDocumentPanel::tryToCloseDocument (Component*)
  126. {
  127. // If you hit this assertion then you need to implement this method in a subclass.
  128. jassertfalse;
  129. return false;
  130. }
  131. #endif
  132. MultiDocumentPanelWindow* MultiDocumentPanel::createNewDocumentWindow()
  133. {
  134. return new MultiDocumentPanelWindow (backgroundColour);
  135. }
  136. void MultiDocumentPanel::addWindow (Component* component)
  137. {
  138. auto* dw = createNewDocumentWindow();
  139. dw->setResizable (true, false);
  140. dw->setContentNonOwned (component, true);
  141. dw->setName (component->getName());
  142. auto bkg = component->getProperties() ["mdiDocumentBkg_"];
  143. dw->setBackgroundColour (bkg.isVoid() ? backgroundColour : Colour ((uint32) static_cast<int> (bkg)));
  144. int x = 4;
  145. if (auto* topComp = getChildren().getLast())
  146. if (topComp->getX() == x && topComp->getY() == x)
  147. x += 16;
  148. dw->setTopLeftPosition (x, x);
  149. auto pos = component->getProperties() ["mdiDocumentPos_"];
  150. if (pos.toString().isNotEmpty())
  151. dw->restoreWindowStateFromString (pos.toString());
  152. addAndMakeVisible (dw);
  153. dw->toFront (true);
  154. }
  155. bool MultiDocumentPanel::addDocument (Component* const component,
  156. Colour docColour,
  157. const bool deleteWhenRemoved)
  158. {
  159. // If you try passing a full DocumentWindow or ResizableWindow in here, you'll end up
  160. // with a frame-within-a-frame! Just pass in the bare content component.
  161. jassert (dynamic_cast<ResizableWindow*> (component) == nullptr);
  162. if (component == nullptr || (maximumNumDocuments > 0 && components.size() >= maximumNumDocuments))
  163. return false;
  164. components.add (component);
  165. component->getProperties().set ("mdiDocumentDelete_", deleteWhenRemoved);
  166. component->getProperties().set ("mdiDocumentBkg_", (int) docColour.getARGB());
  167. component->addComponentListener (this);
  168. if (mode == FloatingWindows)
  169. {
  170. if (isFullscreenWhenOneDocument())
  171. {
  172. if (components.size() == 1)
  173. {
  174. addAndMakeVisible (component);
  175. }
  176. else
  177. {
  178. if (components.size() == 2)
  179. addWindow (components.getFirst());
  180. addWindow (component);
  181. }
  182. }
  183. else
  184. {
  185. addWindow (component);
  186. }
  187. }
  188. else
  189. {
  190. if (tabComponent == nullptr && components.size() > numDocsBeforeTabsUsed)
  191. {
  192. tabComponent.reset (new TabbedComponentInternal());
  193. addAndMakeVisible (tabComponent.get());
  194. auto temp = components;
  195. for (auto& c : temp)
  196. tabComponent->addTab (c->getName(), docColour, c, false);
  197. resized();
  198. }
  199. else
  200. {
  201. if (tabComponent != nullptr)
  202. tabComponent->addTab (component->getName(), docColour, component, false);
  203. else
  204. addAndMakeVisible (component);
  205. }
  206. setActiveDocument (component);
  207. }
  208. resized();
  209. activeDocumentChanged();
  210. return true;
  211. }
  212. void MultiDocumentPanel::closeDocumentInternal (Component* component)
  213. {
  214. // Intellisense warns about component being uninitialised.
  215. // I'm not sure how a function argument could be uninitialised.
  216. JUCE_BEGIN_IGNORE_WARNINGS_MSVC (6001)
  217. component->removeComponentListener (this);
  218. const bool shouldDelete = MultiDocHelpers::shouldDeleteComp (component);
  219. component->getProperties().remove ("mdiDocumentDelete_");
  220. component->getProperties().remove ("mdiDocumentBkg_");
  221. if (mode == FloatingWindows)
  222. {
  223. for (auto* child : getChildren())
  224. {
  225. if (auto* dw = dynamic_cast<MultiDocumentPanelWindow*> (child))
  226. {
  227. if (dw->getContentComponent() == component)
  228. {
  229. std::unique_ptr<MultiDocumentPanelWindow> (dw)->clearContentComponent();
  230. break;
  231. }
  232. }
  233. }
  234. if (shouldDelete)
  235. delete component;
  236. components.removeFirstMatchingValue (component);
  237. if (isFullscreenWhenOneDocument() && components.size() == 1)
  238. {
  239. for (int i = getNumChildComponents(); --i >= 0;)
  240. {
  241. std::unique_ptr<MultiDocumentPanelWindow> dw (dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)));
  242. if (dw != nullptr)
  243. dw->clearContentComponent();
  244. }
  245. addAndMakeVisible (components.getFirst());
  246. }
  247. }
  248. else
  249. {
  250. jassert (components.indexOf (component) >= 0);
  251. if (tabComponent != nullptr)
  252. {
  253. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  254. if (tabComponent->getTabContentComponent (i) == component)
  255. tabComponent->removeTab (i);
  256. }
  257. else
  258. {
  259. removeChildComponent (component);
  260. }
  261. if (shouldDelete)
  262. delete component;
  263. if (tabComponent != nullptr && tabComponent->getNumTabs() <= numDocsBeforeTabsUsed)
  264. tabComponent.reset();
  265. components.removeFirstMatchingValue (component);
  266. if (components.size() > 0 && tabComponent == nullptr)
  267. addAndMakeVisible (components.getFirst());
  268. }
  269. resized();
  270. // This ensures that the active tab is painted properly when a tab is closed!
  271. if (auto* activeComponent = getActiveDocument())
  272. setActiveDocument (activeComponent);
  273. activeDocumentChanged();
  274. JUCE_END_IGNORE_WARNINGS_MSVC
  275. }
  276. #if JUCE_MODAL_LOOPS_PERMITTED
  277. bool MultiDocumentPanel::closeDocument (Component* component,
  278. const bool checkItsOkToCloseFirst)
  279. {
  280. // Intellisense warns about component being uninitialised.
  281. // I'm not sure how a function argument could be uninitialised.
  282. JUCE_BEGIN_IGNORE_WARNINGS_MSVC (6001)
  283. if (component == nullptr)
  284. return true;
  285. if (components.contains (component))
  286. {
  287. if (checkItsOkToCloseFirst && ! tryToCloseDocument (component))
  288. return false;
  289. closeDocumentInternal (component);
  290. }
  291. else
  292. {
  293. jassertfalse;
  294. }
  295. return true;
  296. JUCE_END_IGNORE_WARNINGS_MSVC
  297. }
  298. #endif
  299. void MultiDocumentPanel::closeDocumentAsync (Component* component,
  300. const bool checkItsOkToCloseFirst,
  301. std::function<void (bool)> callback)
  302. {
  303. // Intellisense warns about component being uninitialised.
  304. // I'm not sure how a function argument could be uninitialised.
  305. JUCE_BEGIN_IGNORE_WARNINGS_MSVC (6001)
  306. if (component == nullptr)
  307. {
  308. if (callback != nullptr)
  309. callback (true);
  310. return;
  311. }
  312. if (components.contains (component))
  313. {
  314. if (checkItsOkToCloseFirst)
  315. {
  316. tryToCloseDocumentAsync (component,
  317. [parent = SafePointer<MultiDocumentPanel> { this }, component, callback] (bool closedSuccessfully)
  318. {
  319. if (parent == nullptr)
  320. return;
  321. if (closedSuccessfully)
  322. parent->closeDocumentInternal (component);
  323. if (callback != nullptr)
  324. callback (closedSuccessfully);
  325. });
  326. return;
  327. }
  328. }
  329. else
  330. {
  331. jassertfalse;
  332. }
  333. if (callback != nullptr)
  334. callback (true);
  335. JUCE_END_IGNORE_WARNINGS_MSVC
  336. }
  337. int MultiDocumentPanel::getNumDocuments() const noexcept
  338. {
  339. return components.size();
  340. }
  341. Component* MultiDocumentPanel::getDocument (const int index) const noexcept
  342. {
  343. return components [index];
  344. }
  345. Component* MultiDocumentPanel::getActiveDocument() const noexcept
  346. {
  347. if (mode == FloatingWindows)
  348. {
  349. for (auto* child : getChildren())
  350. if (auto* dw = dynamic_cast<MultiDocumentPanelWindow*> (child))
  351. if (dw->isActiveWindow())
  352. return dw->getContentComponent();
  353. }
  354. return components.getLast();
  355. }
  356. void MultiDocumentPanel::setActiveDocument (Component* component)
  357. {
  358. jassert (component != nullptr);
  359. if (mode == FloatingWindows)
  360. {
  361. component = getContainerComp (component);
  362. if (component != nullptr)
  363. component->toFront (true);
  364. }
  365. else if (tabComponent != nullptr)
  366. {
  367. jassert (components.indexOf (component) >= 0);
  368. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  369. {
  370. if (tabComponent->getTabContentComponent (i) == component)
  371. {
  372. tabComponent->setCurrentTabIndex (i);
  373. break;
  374. }
  375. }
  376. }
  377. else
  378. {
  379. component->grabKeyboardFocus();
  380. }
  381. }
  382. void MultiDocumentPanel::activeDocumentChanged()
  383. {
  384. }
  385. void MultiDocumentPanel::setMaximumNumDocuments (const int newNumber)
  386. {
  387. maximumNumDocuments = newNumber;
  388. }
  389. void MultiDocumentPanel::useFullscreenWhenOneDocument (const bool shouldUseTabs)
  390. {
  391. numDocsBeforeTabsUsed = shouldUseTabs ? 1 : 0;
  392. }
  393. bool MultiDocumentPanel::isFullscreenWhenOneDocument() const noexcept
  394. {
  395. return numDocsBeforeTabsUsed != 0;
  396. }
  397. //==============================================================================
  398. void MultiDocumentPanel::setLayoutMode (const LayoutMode newLayoutMode)
  399. {
  400. if (mode != newLayoutMode)
  401. {
  402. mode = newLayoutMode;
  403. if (mode == FloatingWindows)
  404. {
  405. tabComponent.reset();
  406. }
  407. else
  408. {
  409. for (int i = getNumChildComponents(); --i >= 0;)
  410. {
  411. std::unique_ptr<MultiDocumentPanelWindow> dw (dynamic_cast<MultiDocumentPanelWindow*> (getChildComponent (i)));
  412. if (dw != nullptr)
  413. {
  414. dw->getContentComponent()->getProperties().set ("mdiDocumentPos_", dw->getWindowStateAsString());
  415. dw->clearContentComponent();
  416. }
  417. }
  418. }
  419. resized();
  420. auto tempComps = components;
  421. components.clear();
  422. for (auto* c : tempComps)
  423. addDocument (c,
  424. Colour ((uint32) static_cast<int> (c->getProperties().getWithDefault ("mdiDocumentBkg_", (int) Colours::white.getARGB()))),
  425. MultiDocHelpers::shouldDeleteComp (c));
  426. }
  427. }
  428. void MultiDocumentPanel::setBackgroundColour (Colour newBackgroundColour)
  429. {
  430. if (backgroundColour != newBackgroundColour)
  431. {
  432. backgroundColour = newBackgroundColour;
  433. setOpaque (newBackgroundColour.isOpaque());
  434. repaint();
  435. }
  436. }
  437. //==============================================================================
  438. void MultiDocumentPanel::paint (Graphics& g)
  439. {
  440. g.fillAll (backgroundColour);
  441. }
  442. void MultiDocumentPanel::resized()
  443. {
  444. if (mode == MaximisedWindowsWithTabs || components.size() == numDocsBeforeTabsUsed)
  445. {
  446. for (auto* child : getChildren())
  447. child->setBounds (getLocalBounds());
  448. }
  449. setWantsKeyboardFocus (components.size() == 0);
  450. }
  451. Component* MultiDocumentPanel::getContainerComp (Component* c) const
  452. {
  453. if (mode == FloatingWindows)
  454. {
  455. for (auto* child : getChildren())
  456. if (auto* dw = dynamic_cast<MultiDocumentPanelWindow*> (child))
  457. if (dw->getContentComponent() == c)
  458. return dw;
  459. }
  460. return c;
  461. }
  462. void MultiDocumentPanel::componentNameChanged (Component&)
  463. {
  464. if (mode == FloatingWindows)
  465. {
  466. for (auto* child : getChildren())
  467. if (auto* dw = dynamic_cast<MultiDocumentPanelWindow*> (child))
  468. dw->setName (dw->getContentComponent()->getName());
  469. }
  470. else if (tabComponent != nullptr)
  471. {
  472. for (int i = tabComponent->getNumTabs(); --i >= 0;)
  473. tabComponent->setTabName (i, tabComponent->getTabContentComponent (i)->getName());
  474. }
  475. }
  476. void MultiDocumentPanel::updateOrder()
  477. {
  478. auto oldList = components;
  479. if (mode == FloatingWindows)
  480. {
  481. components.clear();
  482. for (auto* child : getChildren())
  483. if (auto* dw = dynamic_cast<MultiDocumentPanelWindow*> (child))
  484. components.add (dw->getContentComponent());
  485. }
  486. else
  487. {
  488. if (tabComponent != nullptr)
  489. {
  490. if (auto* current = tabComponent->getCurrentContentComponent())
  491. {
  492. components.removeFirstMatchingValue (current);
  493. components.add (current);
  494. }
  495. }
  496. }
  497. if (components != oldList)
  498. activeDocumentChanged();
  499. }
  500. } // namespace juce