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.

611 lines
17KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2015 - ROLI Ltd.
  5. Permission is granted to use this software under the terms of either:
  6. a) the GPL v2 (or any later version)
  7. b) the Affero GPL v3
  8. Details of these licenses can be found at: www.gnu.org/licenses
  9. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  10. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  11. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  12. ------------------------------------------------------------------------------
  13. To release a closed-source product which uses JUCE, commercial licenses are
  14. available: visit www.juce.com for more information.
  15. ==============================================================================
  16. */
  17. #include "../jucer_Headers.h"
  18. #include "jucer_Application.h"
  19. #include "jucer_MainWindow.h"
  20. #include "jucer_OpenDocumentManager.h"
  21. #include "../Code Editor/jucer_SourceCodeEditor.h"
  22. #include "../Wizards/jucer_NewProjectWizardClasses.h"
  23. #include "../Utility/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. setResizeLimits (600, 500, 32000, 32000);
  55. }
  56. MainWindow::~MainWindow()
  57. {
  58. #if ! JUCE_MAC
  59. setMenuBar (nullptr);
  60. #endif
  61. removeKeyListener (ProjucerApplication::getCommandManager().getKeyMappings());
  62. // save the current size and position to our settings file..
  63. getGlobalProperties().setValue ("lastMainWindowPos", getWindowStateAsString());
  64. clearContentComponent();
  65. currentProject = nullptr;
  66. }
  67. void MainWindow::createProjectContentCompIfNeeded()
  68. {
  69. if (getProjectContentComponent() == nullptr)
  70. {
  71. clearContentComponent();
  72. setContentOwned (new ProjectContentComponent(), false);
  73. jassert (getProjectContentComponent() != nullptr);
  74. }
  75. }
  76. void MainWindow::makeVisible()
  77. {
  78. restoreWindowPosition();
  79. setVisible (true);
  80. addToDesktop(); // (must add before restoring size so that fullscreen will work)
  81. restoreWindowPosition();
  82. getContentComponent()->grabKeyboardFocus();
  83. }
  84. ProjectContentComponent* MainWindow::getProjectContentComponent() const
  85. {
  86. return dynamic_cast<ProjectContentComponent*> (getContentComponent());
  87. }
  88. void MainWindow::closeButtonPressed()
  89. {
  90. ProjucerApplication::getApp().mainWindowList.closeWindow (this);
  91. }
  92. bool MainWindow::closeProject (Project* project)
  93. {
  94. jassert (project == currentProject && project != nullptr);
  95. if (project == nullptr)
  96. return true;
  97. project->getStoredProperties().setValue (getProjectWindowPosName(), getWindowStateAsString());
  98. if (ProjectContentComponent* const pcc = getProjectContentComponent())
  99. {
  100. pcc->saveTreeViewState();
  101. pcc->saveOpenDocumentList();
  102. pcc->hideEditor();
  103. }
  104. if (! ProjucerApplication::getApp().openDocumentManager.closeAllDocumentsUsingProject (*project, true))
  105. return false;
  106. FileBasedDocument::SaveResult r = project->saveIfNeededAndUserAgrees();
  107. if (r == FileBasedDocument::savedOk)
  108. {
  109. setProject (nullptr);
  110. return true;
  111. }
  112. return false;
  113. }
  114. bool MainWindow::closeCurrentProject()
  115. {
  116. return currentProject == nullptr || closeProject (currentProject);
  117. }
  118. void MainWindow::setProject (Project* newProject)
  119. {
  120. createProjectContentCompIfNeeded();
  121. getProjectContentComponent()->setProject (newProject);
  122. currentProject = newProject;
  123. ProjucerApplication::getCommandManager().commandStatusChanged();
  124. }
  125. void MainWindow::restoreWindowPosition()
  126. {
  127. String windowState;
  128. if (currentProject != nullptr)
  129. windowState = currentProject->getStoredProperties().getValue (getProjectWindowPosName());
  130. if (windowState.isEmpty())
  131. windowState = getGlobalProperties().getValue ("lastMainWindowPos");
  132. restoreWindowStateFromString (windowState);
  133. }
  134. bool MainWindow::canOpenFile (const File& file) const
  135. {
  136. return (! file.isDirectory())
  137. && (file.hasFileExtension (Project::projectFileExtension)
  138. || ProjucerApplication::getApp().openDocumentManager.canOpenFile (file));
  139. }
  140. bool MainWindow::openFile (const File& file)
  141. {
  142. createProjectContentCompIfNeeded();
  143. if (file.hasFileExtension (Project::projectFileExtension))
  144. {
  145. ScopedPointer<Project> newDoc (new Project (file));
  146. Result result (newDoc->loadFrom (file, true));
  147. if (result.wasOk() && closeCurrentProject())
  148. {
  149. setProject (newDoc);
  150. newDoc.release()->setChangedFlag (false);
  151. jassert (getProjectContentComponent() != nullptr);
  152. getProjectContentComponent()->reloadLastOpenDocuments();
  153. if (Project* p = getProject())
  154. p->updateDeprecatedProjectSettingsInteractively();
  155. return true;
  156. }
  157. }
  158. else if (file.exists())
  159. {
  160. return getProjectContentComponent()->showEditorForFile (file, true);
  161. }
  162. return false;
  163. }
  164. bool MainWindow::isInterestedInFileDrag (const StringArray& filenames)
  165. {
  166. for (int i = filenames.size(); --i >= 0;)
  167. if (canOpenFile (File (filenames[i])))
  168. return true;
  169. return false;
  170. }
  171. void MainWindow::filesDropped (const StringArray& filenames, int /*mouseX*/, int /*mouseY*/)
  172. {
  173. for (int i = filenames.size(); --i >= 0;)
  174. {
  175. const File f (filenames[i]);
  176. if (canOpenFile (f) && openFile (f))
  177. break;
  178. }
  179. }
  180. bool MainWindow::shouldDropFilesWhenDraggedExternally (const DragAndDropTarget::SourceDetails& sourceDetails,
  181. StringArray& files, bool& canMoveFiles)
  182. {
  183. if (TreeView* tv = dynamic_cast<TreeView*> (sourceDetails.sourceComponent.get()))
  184. {
  185. Array<JucerTreeViewBase*> selected;
  186. for (int i = tv->getNumSelectedItems(); --i >= 0;)
  187. if (JucerTreeViewBase* b = dynamic_cast<JucerTreeViewBase*> (tv->getSelectedItem(i)))
  188. selected.add (b);
  189. if (selected.size() > 0)
  190. {
  191. for (int i = selected.size(); --i >= 0;)
  192. {
  193. if (JucerTreeViewBase* jtvb = selected.getUnchecked(i))
  194. {
  195. const File f (jtvb->getDraggableFile());
  196. if (f.existsAsFile())
  197. files.add (f.getFullPathName());
  198. }
  199. }
  200. canMoveFiles = false;
  201. return files.size() > 0;
  202. }
  203. }
  204. return false;
  205. }
  206. void MainWindow::activeWindowStatusChanged()
  207. {
  208. DocumentWindow::activeWindowStatusChanged();
  209. if (ProjectContentComponent* const pcc = getProjectContentComponent())
  210. pcc->updateMissingFileStatuses();
  211. ProjucerApplication::getApp().openDocumentManager.reloadModifiedFiles();
  212. if (Project* p = getProject())
  213. {
  214. if (p->hasProjectBeenModified())
  215. {
  216. const int r = AlertWindow::showOkCancelBox (AlertWindow::QuestionIcon,
  217. TRANS ("The .jucer file has been modified since the last save."),
  218. TRANS ("Do you want to keep the current project or re-load from disk?"),
  219. TRANS ("Keep"),
  220. TRANS ("Re-load from disk"));
  221. if (r == 0)
  222. {
  223. File projectFile = p->getFile();
  224. setProject (nullptr);
  225. openFile (projectFile);
  226. }
  227. else if (r == 1)
  228. {
  229. ProjucerApplication::getApp().getCommandManager().invokeDirectly (CommandIDs::saveProject, true);
  230. }
  231. }
  232. }
  233. }
  234. void MainWindow::showNewProjectWizard()
  235. {
  236. jassert (currentProject == nullptr);
  237. setContentOwned (createNewProjectWizardComponent(), true);
  238. centreWithSize (900, 630);
  239. setVisible (true);
  240. addToDesktop();
  241. getContentComponent()->grabKeyboardFocus();
  242. }
  243. //==============================================================================
  244. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  245. {
  246. return nullptr;
  247. }
  248. void MainWindow::getAllCommands (Array <CommandID>& commands)
  249. {
  250. const CommandID ids[] = { CommandIDs::closeWindow };
  251. commands.addArray (ids, numElementsInArray (ids));
  252. }
  253. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  254. {
  255. switch (commandID)
  256. {
  257. case CommandIDs::closeWindow:
  258. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  259. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  260. break;
  261. default:
  262. break;
  263. }
  264. }
  265. bool MainWindow::perform (const InvocationInfo& info)
  266. {
  267. switch (info.commandID)
  268. {
  269. case CommandIDs::closeWindow:
  270. closeButtonPressed();
  271. break;
  272. default:
  273. return false;
  274. }
  275. return true;
  276. }
  277. //==============================================================================
  278. MainWindowList::MainWindowList()
  279. {
  280. }
  281. void MainWindowList::forceCloseAllWindows()
  282. {
  283. windows.clear();
  284. }
  285. bool MainWindowList::askAllWindowsToClose()
  286. {
  287. saveCurrentlyOpenProjectList();
  288. while (windows.size() > 0)
  289. {
  290. if (! windows[0]->closeCurrentProject())
  291. return false;
  292. windows.remove (0);
  293. }
  294. return true;
  295. }
  296. void MainWindowList::createWindowIfNoneAreOpen()
  297. {
  298. if (windows.size() == 0)
  299. createNewMainWindow()->showNewProjectWizard();
  300. }
  301. void MainWindowList::closeWindow (MainWindow* w)
  302. {
  303. jassert (windows.contains (w));
  304. #if ! JUCE_MAC
  305. if (windows.size() == 1)
  306. {
  307. JUCEApplicationBase::getInstance()->systemRequestedQuit();
  308. }
  309. else
  310. #endif
  311. {
  312. if (w->closeCurrentProject())
  313. {
  314. windows.removeObject (w);
  315. saveCurrentlyOpenProjectList();
  316. }
  317. }
  318. }
  319. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  320. {
  321. Desktop& desktop = Desktop::getInstance();
  322. for (int i = desktop.getNumComponents(); --i >= 0;)
  323. {
  324. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  325. {
  326. if (ProjectContentComponent* pcc = mw->getProjectContentComponent())
  327. {
  328. if (pcc->hasFileInRecentList (doc->getFile()))
  329. {
  330. mw->toFront (true);
  331. mw->getProjectContentComponent()->showDocument (doc, grabFocus);
  332. return;
  333. }
  334. }
  335. }
  336. }
  337. getFrontmostWindow()->getProjectContentComponent()->showDocument (doc, grabFocus);
  338. }
  339. bool MainWindowList::openFile (const File& file)
  340. {
  341. for (int i = windows.size(); --i >= 0;)
  342. {
  343. MainWindow* const w = windows.getUnchecked(i);
  344. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  345. {
  346. w->toFront (true);
  347. return true;
  348. }
  349. }
  350. if (file.hasFileExtension (Project::projectFileExtension))
  351. {
  352. MainWindow* const w = getOrCreateEmptyWindow();
  353. bool ok = w->openFile (file);
  354. w->makeVisible();
  355. avoidSuperimposedWindows (w);
  356. return ok;
  357. }
  358. if (file.exists())
  359. return getFrontmostWindow()->openFile (file);
  360. return false;
  361. }
  362. MainWindow* MainWindowList::createNewMainWindow()
  363. {
  364. MainWindow* const w = new MainWindow();
  365. windows.add (w);
  366. w->restoreWindowPosition();
  367. avoidSuperimposedWindows (w);
  368. return w;
  369. }
  370. MainWindow* MainWindowList::getFrontmostWindow (bool createIfNotFound)
  371. {
  372. if (windows.size() == 0)
  373. {
  374. if (createIfNotFound)
  375. {
  376. MainWindow* w = createNewMainWindow();
  377. avoidSuperimposedWindows (w);
  378. w->makeVisible();
  379. return w;
  380. }
  381. return nullptr;
  382. }
  383. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  384. {
  385. MainWindow* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  386. if (windows.contains (mw))
  387. return mw;
  388. }
  389. return windows.getLast();
  390. }
  391. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  392. {
  393. if (windows.size() == 0)
  394. return createNewMainWindow();
  395. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  396. {
  397. MainWindow* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  398. if (windows.contains (mw) && mw->getProject() == nullptr)
  399. return mw;
  400. }
  401. return createNewMainWindow();
  402. }
  403. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  404. {
  405. for (int i = windows.size(); --i >= 0;)
  406. {
  407. MainWindow* const other = windows.getUnchecked(i);
  408. const Rectangle<int> b1 (mw->getBounds());
  409. const Rectangle<int> b2 (other->getBounds());
  410. if (mw != other
  411. && std::abs (b1.getX() - b2.getX()) < 3
  412. && std::abs (b1.getY() - b2.getY()) < 3
  413. && std::abs (b1.getRight() - b2.getRight()) < 3
  414. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  415. {
  416. int dx = 40, dy = 30;
  417. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  418. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  419. mw->setBounds (b1.translated (dx, dy));
  420. }
  421. }
  422. }
  423. void MainWindowList::saveCurrentlyOpenProjectList()
  424. {
  425. Array<File> projects;
  426. Desktop& desktop = Desktop::getInstance();
  427. for (int i = 0; i < desktop.getNumComponents(); ++i)
  428. {
  429. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  430. if (Project* p = mw->getProject())
  431. projects.add (p->getFile());
  432. }
  433. getAppSettings().setLastProjects (projects);
  434. }
  435. void MainWindowList::reopenLastProjects()
  436. {
  437. Array<File> projects (getAppSettings().getLastProjects());
  438. for (int i = 0; i < projects.size(); ++ i)
  439. openFile (projects.getReference(i));
  440. }
  441. void MainWindowList::sendLookAndFeelChange()
  442. {
  443. for (int i = windows.size(); --i >= 0;)
  444. windows.getUnchecked(i)->sendLookAndFeelChange();
  445. }
  446. Project* MainWindowList::getFrontmostProject()
  447. {
  448. Desktop& desktop = Desktop::getInstance();
  449. for (int i = desktop.getNumComponents(); --i >= 0;)
  450. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  451. if (Project* p = mw->getProject())
  452. return p;
  453. return nullptr;
  454. }
  455. File findDefaultModulesFolder (bool mustContainJuceCoreModule)
  456. {
  457. auto& windows = ProjucerApplication::getApp().mainWindowList;
  458. for (int i = windows.windows.size(); --i >= 0;)
  459. {
  460. if (auto* p = windows.windows.getUnchecked (i)->getProject())
  461. {
  462. const File f (EnabledModuleList::findDefaultModulesFolder (*p));
  463. if (isJuceModulesFolder (f) || (f.isDirectory() && ! mustContainJuceCoreModule))
  464. return f;
  465. }
  466. }
  467. if (mustContainJuceCoreModule)
  468. return findDefaultModulesFolder (false);
  469. auto f = File::getSpecialLocation (File::currentApplicationFile);
  470. for (;;)
  471. {
  472. auto parent = f.getParentDirectory();
  473. if (parent == f || ! parent.isDirectory())
  474. break;
  475. if (isJuceFolder (parent))
  476. return parent.getChildFile ("modules");
  477. f = parent;
  478. }
  479. return {};
  480. }