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.

641 lines
18KB

  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. #include "../Application/jucer_Headers.h"
  20. #include "jucer_Application.h"
  21. #include "jucer_MainWindow.h"
  22. #include "../Wizards/jucer_NewProjectWizardClasses.h"
  23. #include "../Utility/UI/jucer_JucerTreeViewBase.h"
  24. //==============================================================================
  25. MainWindow::MainWindow()
  26. : DocumentWindow (ProjucerApplication::getApp().getApplicationName(),
  27. ProjucerApplication::getApp().lookAndFeel.getCurrentColourScheme()
  28. .getUIColour (LookAndFeel_V4::ColourScheme::UIColour::windowBackground),
  29. DocumentWindow::allButtons,
  30. false)
  31. {
  32. setUsingNativeTitleBar (true);
  33. #if ! JUCE_MAC
  34. setMenuBar (ProjucerApplication::getApp().getMenuModel());
  35. #endif
  36. createProjectContentCompIfNeeded();
  37. setResizable (true, false);
  38. centreWithSize (800, 600);
  39. ApplicationCommandManager& commandManager = ProjucerApplication::getCommandManager();
  40. // Register all the app commands..
  41. commandManager.registerAllCommandsForTarget (this);
  42. commandManager.registerAllCommandsForTarget (getProjectContentComponent());
  43. // update key mappings..
  44. {
  45. commandManager.getKeyMappings()->resetToDefaultMappings();
  46. ScopedPointer<XmlElement> keys (getGlobalProperties().getXmlValue ("keyMappings"));
  47. if (keys != nullptr)
  48. commandManager.getKeyMappings()->restoreFromXml (*keys);
  49. addKeyListener (commandManager.getKeyMappings());
  50. }
  51. // don't want the window to take focus when the title-bar is clicked..
  52. setWantsKeyboardFocus (false);
  53. getLookAndFeel().setColour (ColourSelector::backgroundColourId, Colours::transparentBlack);
  54. projectNameValue.addListener (this);
  55. setResizeLimits (600, 500, 32000, 32000);
  56. }
  57. MainWindow::~MainWindow()
  58. {
  59. #if ! JUCE_MAC
  60. setMenuBar (nullptr);
  61. #endif
  62. removeKeyListener (ProjucerApplication::getCommandManager().getKeyMappings());
  63. // save the current size and position to our settings file..
  64. getGlobalProperties().setValue ("lastMainWindowPos", getWindowStateAsString());
  65. clearContentComponent();
  66. currentProject.reset();
  67. }
  68. void MainWindow::createProjectContentCompIfNeeded()
  69. {
  70. if (getProjectContentComponent() == nullptr)
  71. {
  72. clearContentComponent();
  73. setContentOwned (new ProjectContentComponent(), false);
  74. jassert (getProjectContentComponent() != nullptr);
  75. }
  76. }
  77. void MainWindow::makeVisible()
  78. {
  79. restoreWindowPosition();
  80. setVisible (true);
  81. addToDesktop(); // (must add before restoring size so that fullscreen will work)
  82. restoreWindowPosition();
  83. getContentComponent()->grabKeyboardFocus();
  84. }
  85. ProjectContentComponent* MainWindow::getProjectContentComponent() const
  86. {
  87. return dynamic_cast<ProjectContentComponent*> (getContentComponent());
  88. }
  89. void MainWindow::closeButtonPressed()
  90. {
  91. ProjucerApplication::getApp().mainWindowList.closeWindow (this);
  92. }
  93. bool MainWindow::closeProject (Project* project)
  94. {
  95. jassert (project == currentProject && project != nullptr);
  96. if (project == nullptr)
  97. return true;
  98. project->getStoredProperties().setValue (getProjectWindowPosName(), getWindowStateAsString());
  99. if (auto* pcc = getProjectContentComponent())
  100. {
  101. pcc->saveTreeViewState();
  102. pcc->saveOpenDocumentList();
  103. pcc->hideEditor();
  104. }
  105. if (! ProjucerApplication::getApp().openDocumentManager.closeAllDocumentsUsingProject (*project, true))
  106. return false;
  107. auto r = project->saveIfNeededAndUserAgrees();
  108. if (r == FileBasedDocument::savedOk)
  109. {
  110. setProject (nullptr);
  111. return true;
  112. }
  113. return false;
  114. }
  115. bool MainWindow::closeCurrentProject()
  116. {
  117. return currentProject == nullptr || closeProject (currentProject.get());
  118. }
  119. void MainWindow::setProject (Project* newProject)
  120. {
  121. createProjectContentCompIfNeeded();
  122. getProjectContentComponent()->setProject (newProject);
  123. currentProject.reset (newProject);
  124. if (currentProject != nullptr)
  125. projectNameValue.referTo (currentProject->getProjectValue (Ids::name));
  126. else
  127. projectNameValue.referTo (Value());
  128. ProjucerApplication::getCommandManager().commandStatusChanged();
  129. }
  130. void MainWindow::restoreWindowPosition()
  131. {
  132. String windowState;
  133. if (currentProject != nullptr)
  134. windowState = currentProject->getStoredProperties().getValue (getProjectWindowPosName());
  135. if (windowState.isEmpty())
  136. windowState = getGlobalProperties().getValue ("lastMainWindowPos");
  137. restoreWindowStateFromString (windowState);
  138. }
  139. bool MainWindow::canOpenFile (const File& file) const
  140. {
  141. return (! file.isDirectory())
  142. && (file.hasFileExtension (Project::projectFileExtension)
  143. || ProjucerApplication::getApp().openDocumentManager.canOpenFile (file));
  144. }
  145. bool MainWindow::openFile (const File& file)
  146. {
  147. createProjectContentCompIfNeeded();
  148. if (file.hasFileExtension (Project::projectFileExtension))
  149. {
  150. ScopedPointer<Project> newDoc (new Project (file));
  151. auto result = newDoc->loadFrom (file, true);
  152. if (result.wasOk() && closeCurrentProject())
  153. {
  154. setProject (newDoc.get());
  155. newDoc.release()->setChangedFlag (false);
  156. jassert (getProjectContentComponent() != nullptr);
  157. getProjectContentComponent()->reloadLastOpenDocuments();
  158. if (auto* p = getProject())
  159. p->updateDeprecatedProjectSettingsInteractively();
  160. return true;
  161. }
  162. }
  163. else if (file.exists())
  164. {
  165. return getProjectContentComponent()->showEditorForFile (file, true);
  166. }
  167. return false;
  168. }
  169. bool MainWindow::isInterestedInFileDrag (const StringArray& filenames)
  170. {
  171. for (auto& filename : filenames)
  172. if (canOpenFile (File (filename)))
  173. return true;
  174. return false;
  175. }
  176. void MainWindow::filesDropped (const StringArray& filenames, int /*mouseX*/, int /*mouseY*/)
  177. {
  178. for (auto& filename : filenames)
  179. {
  180. const File f (filename);
  181. if (canOpenFile (f) && openFile (f))
  182. break;
  183. }
  184. }
  185. bool MainWindow::shouldDropFilesWhenDraggedExternally (const DragAndDropTarget::SourceDetails& sourceDetails,
  186. StringArray& files, bool& canMoveFiles)
  187. {
  188. if (auto* tv = dynamic_cast<TreeView*> (sourceDetails.sourceComponent.get()))
  189. {
  190. Array<JucerTreeViewBase*> selected;
  191. for (int i = tv->getNumSelectedItems(); --i >= 0;)
  192. if (auto* b = dynamic_cast<JucerTreeViewBase*> (tv->getSelectedItem(i)))
  193. selected.add (b);
  194. if (! selected.isEmpty())
  195. {
  196. for (int i = selected.size(); --i >= 0;)
  197. {
  198. if (auto* jtvb = selected.getUnchecked(i))
  199. {
  200. auto f = jtvb->getDraggableFile();
  201. if (f.existsAsFile())
  202. files.add (f.getFullPathName());
  203. }
  204. }
  205. canMoveFiles = false;
  206. return ! files.isEmpty();
  207. }
  208. }
  209. return false;
  210. }
  211. void MainWindow::activeWindowStatusChanged()
  212. {
  213. DocumentWindow::activeWindowStatusChanged();
  214. if (auto* pcc = getProjectContentComponent())
  215. pcc->updateMissingFileStatuses();
  216. ProjucerApplication::getApp().openDocumentManager.reloadModifiedFiles();
  217. if (auto* p = getProject())
  218. {
  219. if (p->hasProjectBeenModified())
  220. {
  221. Component::SafePointer<Component> safePointer (this);
  222. MessageManager::callAsync ([=] ()
  223. {
  224. if (safePointer == nullptr)
  225. return; // bail out if the window has been deleted
  226. auto result = AlertWindow::showOkCancelBox (AlertWindow::QuestionIcon,
  227. TRANS ("The .jucer file has been modified since the last save."),
  228. TRANS ("Do you want to keep the current project or re-load from disk?"),
  229. TRANS ("Keep"),
  230. TRANS ("Re-load from disk"));
  231. if (safePointer == nullptr)
  232. return;
  233. if (result == 0)
  234. {
  235. if (auto* project = getProject())
  236. {
  237. auto projectFile = project->getFile();
  238. setProject (nullptr);
  239. openFile (projectFile);
  240. }
  241. }
  242. else
  243. {
  244. ProjucerApplication::getApp().getCommandManager().invokeDirectly (CommandIDs::saveProject, true);
  245. }
  246. });
  247. }
  248. }
  249. }
  250. void MainWindow::showNewProjectWizard()
  251. {
  252. jassert (currentProject == nullptr);
  253. setContentOwned (createNewProjectWizardComponent(), true);
  254. centreWithSize (900, 630);
  255. setVisible (true);
  256. addToDesktop();
  257. getContentComponent()->grabKeyboardFocus();
  258. }
  259. //==============================================================================
  260. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  261. {
  262. return nullptr;
  263. }
  264. void MainWindow::getAllCommands (Array <CommandID>& commands)
  265. {
  266. const CommandID ids[] =
  267. {
  268. CommandIDs::closeWindow,
  269. CommandIDs::goToPreviousWindow,
  270. CommandIDs::goToNextWindow
  271. };
  272. commands.addArray (ids, numElementsInArray (ids));
  273. }
  274. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  275. {
  276. switch (commandID)
  277. {
  278. case CommandIDs::closeWindow:
  279. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  280. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  281. break;
  282. case CommandIDs::goToPreviousWindow:
  283. result.setInfo ("Previous Window", "Activates the previous window", CommandCategories::general, 0);
  284. result.setActive (ProjucerApplication::getApp().mainWindowList.windows.size() > 1);
  285. result.defaultKeypresses.add (KeyPress (KeyPress::tabKey, ModifierKeys::shiftModifier | ModifierKeys::ctrlModifier, 0));
  286. break;
  287. case CommandIDs::goToNextWindow:
  288. result.setInfo ("Next Window", "Activates the next window", CommandCategories::general, 0);
  289. result.setActive (ProjucerApplication::getApp().mainWindowList.windows.size() > 1);
  290. result.defaultKeypresses.add (KeyPress (KeyPress::tabKey, ModifierKeys::ctrlModifier, 0));
  291. break;
  292. default:
  293. break;
  294. }
  295. }
  296. bool MainWindow::perform (const InvocationInfo& info)
  297. {
  298. switch (info.commandID)
  299. {
  300. case CommandIDs::closeWindow:
  301. closeButtonPressed();
  302. break;
  303. case CommandIDs::goToPreviousWindow:
  304. ProjucerApplication::getApp().mainWindowList.goToSiblingWindow (this, -1);
  305. break;
  306. case CommandIDs::goToNextWindow:
  307. ProjucerApplication::getApp().mainWindowList.goToSiblingWindow (this, 1);
  308. break;
  309. default:
  310. return false;
  311. }
  312. return true;
  313. }
  314. void MainWindow::valueChanged (Value& v)
  315. {
  316. if (v == Value())
  317. setName ("Projucer");
  318. else
  319. setName (projectNameValue.toString() + " - Projucer");
  320. }
  321. //==============================================================================
  322. MainWindowList::MainWindowList()
  323. {
  324. }
  325. void MainWindowList::forceCloseAllWindows()
  326. {
  327. windows.clear();
  328. }
  329. bool MainWindowList::askAllWindowsToClose()
  330. {
  331. saveCurrentlyOpenProjectList();
  332. while (windows.size() > 0)
  333. {
  334. if (! windows[0]->closeCurrentProject())
  335. return false;
  336. windows.remove (0);
  337. }
  338. return true;
  339. }
  340. void MainWindowList::createWindowIfNoneAreOpen()
  341. {
  342. if (windows.size() == 0)
  343. createNewMainWindow()->showNewProjectWizard();
  344. }
  345. void MainWindowList::closeWindow (MainWindow* w)
  346. {
  347. jassert (windows.contains (w));
  348. #if ! JUCE_MAC
  349. if (windows.size() == 1)
  350. {
  351. JUCEApplicationBase::getInstance()->systemRequestedQuit();
  352. }
  353. else
  354. #endif
  355. {
  356. if (w->closeCurrentProject())
  357. {
  358. windows.removeObject (w);
  359. saveCurrentlyOpenProjectList();
  360. }
  361. }
  362. }
  363. void MainWindowList::goToSiblingWindow (MainWindow* w, int delta)
  364. {
  365. auto index = windows.indexOf (w);
  366. if (index >= 0)
  367. if (auto* next = windows[(index + delta + windows.size()) % windows.size()])
  368. next->toFront (true);
  369. }
  370. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  371. {
  372. auto& desktop = Desktop::getInstance();
  373. for (int i = desktop.getNumComponents(); --i >= 0;)
  374. {
  375. if (auto* mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  376. {
  377. if (auto* pcc = mw->getProjectContentComponent())
  378. {
  379. if (pcc->hasFileInRecentList (doc->getFile()))
  380. {
  381. mw->toFront (true);
  382. mw->getProjectContentComponent()->showDocument (doc, grabFocus);
  383. return;
  384. }
  385. }
  386. }
  387. }
  388. getFrontmostWindow()->getProjectContentComponent()->showDocument (doc, grabFocus);
  389. }
  390. bool MainWindowList::openFile (const File& file, bool openInBackground)
  391. {
  392. for (auto* w : windows)
  393. {
  394. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  395. {
  396. w->toFront (true);
  397. return true;
  398. }
  399. }
  400. if (file.hasFileExtension (Project::projectFileExtension))
  401. {
  402. auto previousFrontWindow = getFrontmostWindow();
  403. auto* w = getOrCreateEmptyWindow();
  404. bool ok = w->openFile (file);
  405. w->makeVisible();
  406. avoidSuperimposedWindows (w);
  407. if (openInBackground && (previousFrontWindow != nullptr))
  408. previousFrontWindow->toFront (true);
  409. return ok;
  410. }
  411. if (file.exists())
  412. return getFrontmostWindow()->openFile (file);
  413. return false;
  414. }
  415. MainWindow* MainWindowList::createNewMainWindow()
  416. {
  417. auto w = new MainWindow();
  418. windows.add (w);
  419. w->restoreWindowPosition();
  420. avoidSuperimposedWindows (w);
  421. return w;
  422. }
  423. MainWindow* MainWindowList::getFrontmostWindow (bool createIfNotFound)
  424. {
  425. if (windows.isEmpty())
  426. {
  427. if (createIfNotFound)
  428. {
  429. auto* w = createNewMainWindow();
  430. avoidSuperimposedWindows (w);
  431. w->makeVisible();
  432. return w;
  433. }
  434. return nullptr;
  435. }
  436. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  437. {
  438. auto* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  439. if (windows.contains (mw))
  440. return mw;
  441. }
  442. return windows.getLast();
  443. }
  444. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  445. {
  446. if (windows.size() == 0)
  447. return createNewMainWindow();
  448. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  449. {
  450. auto* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  451. if (windows.contains (mw) && mw->getProject() == nullptr)
  452. return mw;
  453. }
  454. return createNewMainWindow();
  455. }
  456. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  457. {
  458. for (int i = windows.size(); --i >= 0;)
  459. {
  460. auto* other = windows.getUnchecked(i);
  461. auto b1 = mw->getBounds();
  462. auto b2 = other->getBounds();
  463. if (mw != other
  464. && std::abs (b1.getX() - b2.getX()) < 3
  465. && std::abs (b1.getY() - b2.getY()) < 3
  466. && std::abs (b1.getRight() - b2.getRight()) < 3
  467. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  468. {
  469. int dx = 40, dy = 30;
  470. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  471. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  472. mw->setBounds (b1.translated (dx, dy));
  473. }
  474. }
  475. }
  476. void MainWindowList::saveCurrentlyOpenProjectList()
  477. {
  478. Array<File> projects;
  479. auto& desktop = Desktop::getInstance();
  480. for (int i = 0; i < desktop.getNumComponents(); ++i)
  481. {
  482. if (auto* mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  483. if (auto* p = mw->getProject())
  484. projects.add (p->getFile());
  485. }
  486. getAppSettings().setLastProjects (projects);
  487. }
  488. void MainWindowList::reopenLastProjects()
  489. {
  490. for (auto& p : getAppSettings().getLastProjects())
  491. openFile (p, true);
  492. }
  493. void MainWindowList::sendLookAndFeelChange()
  494. {
  495. for (auto* w : windows)
  496. w->sendLookAndFeelChange();
  497. }
  498. Project* MainWindowList::getFrontmostProject()
  499. {
  500. auto& desktop = Desktop::getInstance();
  501. for (int i = desktop.getNumComponents(); --i >= 0;)
  502. if (auto* mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  503. if (auto* p = mw->getProject())
  504. return p;
  505. return nullptr;
  506. }