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.

610 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. Colour::greyLevel (0.6f),
  28. DocumentWindow::allButtons,
  29. false)
  30. {
  31. setUsingNativeTitleBar (true);
  32. #if ! JUCE_MAC
  33. setMenuBar (ProjucerApplication::getApp().getMenuModel());
  34. #endif
  35. createProjectContentCompIfNeeded();
  36. setResizable (true, false);
  37. centreWithSize (800, 600);
  38. ApplicationCommandManager& commandManager = ProjucerApplication::getCommandManager();
  39. // Register all the app commands..
  40. commandManager.registerAllCommandsForTarget (this);
  41. commandManager.registerAllCommandsForTarget (getProjectContentComponent());
  42. // update key mappings..
  43. {
  44. commandManager.getKeyMappings()->resetToDefaultMappings();
  45. ScopedPointer<XmlElement> keys (getGlobalProperties().getXmlValue ("keyMappings"));
  46. if (keys != nullptr)
  47. commandManager.getKeyMappings()->restoreFromXml (*keys);
  48. addKeyListener (commandManager.getKeyMappings());
  49. }
  50. // don't want the window to take focus when the title-bar is clicked..
  51. setWantsKeyboardFocus (false);
  52. getLookAndFeel().setColour (ColourSelector::backgroundColourId, Colours::transparentBlack);
  53. setResizeLimits (600, 500, 32000, 32000);
  54. }
  55. MainWindow::~MainWindow()
  56. {
  57. #if ! JUCE_MAC
  58. setMenuBar (nullptr);
  59. #endif
  60. removeKeyListener (ProjucerApplication::getCommandManager().getKeyMappings());
  61. // save the current size and position to our settings file..
  62. getGlobalProperties().setValue ("lastMainWindowPos", getWindowStateAsString());
  63. clearContentComponent();
  64. currentProject = nullptr;
  65. }
  66. void MainWindow::createProjectContentCompIfNeeded()
  67. {
  68. if (getProjectContentComponent() == nullptr)
  69. {
  70. clearContentComponent();
  71. setContentOwned (new ProjectContentComponent(), false);
  72. jassert (getProjectContentComponent() != nullptr);
  73. }
  74. }
  75. void MainWindow::makeVisible()
  76. {
  77. restoreWindowPosition();
  78. setVisible (true);
  79. addToDesktop(); // (must add before restoring size so that fullscreen will work)
  80. restoreWindowPosition();
  81. getContentComponent()->grabKeyboardFocus();
  82. }
  83. ProjectContentComponent* MainWindow::getProjectContentComponent() const
  84. {
  85. return dynamic_cast<ProjectContentComponent*> (getContentComponent());
  86. }
  87. void MainWindow::closeButtonPressed()
  88. {
  89. ProjucerApplication::getApp().mainWindowList.closeWindow (this);
  90. }
  91. bool MainWindow::closeProject (Project* project)
  92. {
  93. jassert (project == currentProject && project != nullptr);
  94. if (project == nullptr)
  95. return true;
  96. project->getStoredProperties().setValue (getProjectWindowPosName(), getWindowStateAsString());
  97. if (ProjectContentComponent* const pcc = getProjectContentComponent())
  98. {
  99. pcc->saveTreeViewState();
  100. pcc->saveOpenDocumentList();
  101. pcc->hideEditor();
  102. }
  103. if (! ProjucerApplication::getApp().openDocumentManager.closeAllDocumentsUsingProject (*project, true))
  104. return false;
  105. FileBasedDocument::SaveResult r = project->saveIfNeededAndUserAgrees();
  106. if (r == FileBasedDocument::savedOk)
  107. {
  108. setProject (nullptr);
  109. return true;
  110. }
  111. return false;
  112. }
  113. bool MainWindow::closeCurrentProject()
  114. {
  115. return currentProject == nullptr || closeProject (currentProject);
  116. }
  117. void MainWindow::setProject (Project* newProject)
  118. {
  119. createProjectContentCompIfNeeded();
  120. getProjectContentComponent()->setProject (newProject);
  121. currentProject = newProject;
  122. getProjectContentComponent()->updateMainWindowTitle();
  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. }
  213. void MainWindow::updateTitle (const String& documentName)
  214. {
  215. String name (ProjucerApplication::getApp().getApplicationName());
  216. if (currentProject != nullptr)
  217. {
  218. String projectName (currentProject->getDocumentTitle());
  219. if (currentProject->getFile().getFileNameWithoutExtension() != projectName)
  220. projectName = currentProject->getFile().getFileName();
  221. name << " - " << projectName;
  222. }
  223. if (documentName.isNotEmpty())
  224. name << " - " << documentName;
  225. setName (name);
  226. }
  227. void MainWindow::showNewProjectWizard()
  228. {
  229. jassert (currentProject == nullptr);
  230. setContentOwned (createNewProjectWizardComponent(), true);
  231. centreWithSize (900, 630);
  232. setVisible (true);
  233. addToDesktop();
  234. getContentComponent()->grabKeyboardFocus();
  235. }
  236. //==============================================================================
  237. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  238. {
  239. return nullptr;
  240. }
  241. void MainWindow::getAllCommands (Array <CommandID>& commands)
  242. {
  243. const CommandID ids[] = { CommandIDs::closeWindow };
  244. commands.addArray (ids, numElementsInArray (ids));
  245. }
  246. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  247. {
  248. switch (commandID)
  249. {
  250. case CommandIDs::closeWindow:
  251. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  252. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  253. break;
  254. default:
  255. break;
  256. }
  257. }
  258. bool MainWindow::perform (const InvocationInfo& info)
  259. {
  260. switch (info.commandID)
  261. {
  262. case CommandIDs::closeWindow:
  263. closeButtonPressed();
  264. break;
  265. default:
  266. return false;
  267. }
  268. return true;
  269. }
  270. //==============================================================================
  271. MainWindowList::MainWindowList()
  272. {
  273. }
  274. void MainWindowList::forceCloseAllWindows()
  275. {
  276. windows.clear();
  277. }
  278. bool MainWindowList::askAllWindowsToClose()
  279. {
  280. saveCurrentlyOpenProjectList();
  281. while (windows.size() > 0)
  282. {
  283. if (! windows[0]->closeCurrentProject())
  284. return false;
  285. windows.remove (0);
  286. }
  287. return true;
  288. }
  289. void MainWindowList::createWindowIfNoneAreOpen()
  290. {
  291. if (windows.size() == 0)
  292. createNewMainWindow()->showNewProjectWizard();
  293. }
  294. void MainWindowList::closeWindow (MainWindow* w)
  295. {
  296. jassert (windows.contains (w));
  297. #if ! JUCE_MAC
  298. if (windows.size() == 1)
  299. {
  300. JUCEApplicationBase::getInstance()->systemRequestedQuit();
  301. }
  302. else
  303. #endif
  304. {
  305. if (w->closeCurrentProject())
  306. {
  307. windows.removeObject (w);
  308. saveCurrentlyOpenProjectList();
  309. }
  310. }
  311. }
  312. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  313. {
  314. Desktop& desktop = Desktop::getInstance();
  315. for (int i = desktop.getNumComponents(); --i >= 0;)
  316. {
  317. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  318. {
  319. if (ProjectContentComponent* pcc = mw->getProjectContentComponent())
  320. {
  321. if (pcc->hasFileInRecentList (doc->getFile()))
  322. {
  323. mw->toFront (true);
  324. mw->getProjectContentComponent()->showDocument (doc, grabFocus);
  325. return;
  326. }
  327. }
  328. }
  329. }
  330. getOrCreateFrontmostWindow()->getProjectContentComponent()->showDocument (doc, grabFocus);
  331. }
  332. bool MainWindowList::openFile (const File& file)
  333. {
  334. for (int i = windows.size(); --i >= 0;)
  335. {
  336. MainWindow* const w = windows.getUnchecked(i);
  337. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  338. {
  339. w->toFront (true);
  340. return true;
  341. }
  342. }
  343. if (file.hasFileExtension (Project::projectFileExtension))
  344. {
  345. MainWindow* const w = getOrCreateEmptyWindow();
  346. bool ok = w->openFile (file);
  347. w->makeVisible();
  348. avoidSuperimposedWindows (w);
  349. return ok;
  350. }
  351. if (file.exists())
  352. return getOrCreateFrontmostWindow()->openFile (file);
  353. return false;
  354. }
  355. MainWindow* MainWindowList::createNewMainWindow()
  356. {
  357. MainWindow* const w = new MainWindow();
  358. windows.add (w);
  359. w->restoreWindowPosition();
  360. avoidSuperimposedWindows (w);
  361. return w;
  362. }
  363. MainWindow* MainWindowList::getOrCreateFrontmostWindow()
  364. {
  365. if (windows.size() == 0)
  366. {
  367. MainWindow* w = createNewMainWindow();
  368. avoidSuperimposedWindows (w);
  369. w->makeVisible();
  370. return w;
  371. }
  372. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  373. {
  374. MainWindow* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  375. if (windows.contains (mw))
  376. return mw;
  377. }
  378. return windows.getLast();
  379. }
  380. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  381. {
  382. if (windows.size() == 0)
  383. return createNewMainWindow();
  384. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  385. {
  386. MainWindow* mw = dynamic_cast<MainWindow*> (Desktop::getInstance().getComponent (i));
  387. if (windows.contains (mw) && mw->getProject() == nullptr)
  388. return mw;
  389. }
  390. return createNewMainWindow();
  391. }
  392. void MainWindowList::updateAllWindowTitles()
  393. {
  394. for (int i = 0; i < windows.size(); ++i)
  395. if (ProjectContentComponent* pc = windows.getUnchecked(i)->getProjectContentComponent())
  396. pc->updateMainWindowTitle();
  397. }
  398. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  399. {
  400. for (int i = windows.size(); --i >= 0;)
  401. {
  402. MainWindow* const other = windows.getUnchecked(i);
  403. const Rectangle<int> b1 (mw->getBounds());
  404. const Rectangle<int> b2 (other->getBounds());
  405. if (mw != other
  406. && std::abs (b1.getX() - b2.getX()) < 3
  407. && std::abs (b1.getY() - b2.getY()) < 3
  408. && std::abs (b1.getRight() - b2.getRight()) < 3
  409. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  410. {
  411. int dx = 40, dy = 30;
  412. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  413. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  414. mw->setBounds (b1.translated (dx, dy));
  415. }
  416. }
  417. }
  418. void MainWindowList::saveCurrentlyOpenProjectList()
  419. {
  420. Array<File> projects;
  421. Desktop& desktop = Desktop::getInstance();
  422. for (int i = 0; i < desktop.getNumComponents(); ++i)
  423. {
  424. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  425. if (Project* p = mw->getProject())
  426. projects.add (p->getFile());
  427. }
  428. getAppSettings().setLastProjects (projects);
  429. }
  430. void MainWindowList::reopenLastProjects()
  431. {
  432. Array<File> projects (getAppSettings().getLastProjects());
  433. for (int i = 0; i < projects.size(); ++ i)
  434. openFile (projects.getReference(i));
  435. }
  436. void MainWindowList::sendLookAndFeelChange()
  437. {
  438. for (int i = windows.size(); --i >= 0;)
  439. windows.getUnchecked(i)->sendLookAndFeelChange();
  440. }
  441. Project* MainWindowList::getFrontmostProject()
  442. {
  443. Desktop& desktop = Desktop::getInstance();
  444. for (int i = desktop.getNumComponents(); --i >= 0;)
  445. if (MainWindow* const mw = dynamic_cast<MainWindow*> (desktop.getComponent(i)))
  446. if (Project* p = mw->getProject())
  447. return p;
  448. return nullptr;
  449. }
  450. File findDefaultModulesFolder (bool mustContainJuceCoreModule)
  451. {
  452. const MainWindowList& windows = ProjucerApplication::getApp().mainWindowList;
  453. for (int i = windows.windows.size(); --i >= 0;)
  454. {
  455. if (Project* p = windows.windows.getUnchecked (i)->getProject())
  456. {
  457. const File f (EnabledModuleList::findDefaultModulesFolder (*p));
  458. if (isJuceModulesFolder (f) || (f.isDirectory() && ! mustContainJuceCoreModule))
  459. return f;
  460. }
  461. }
  462. if (mustContainJuceCoreModule)
  463. return findDefaultModulesFolder (false);
  464. File f (File::getSpecialLocation (File::currentApplicationFile));
  465. for (;;)
  466. {
  467. File parent (f.getParentDirectory());
  468. if (parent == f || ! parent.isDirectory())
  469. break;
  470. if (isJuceFolder (parent))
  471. return parent.getChildFile ("modules");
  472. f = parent;
  473. }
  474. return File();
  475. }