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.

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