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.

613 lines
17KB

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