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.

665 lines
19KB

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