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.

532 lines
15KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library - "Jules' Utility Class Extensions"
  4. Copyright 2004-11 by Raw Material Software Ltd.
  5. ------------------------------------------------------------------------------
  6. JUCE can be redistributed and/or modified under the terms of the GNU General
  7. Public License (Version 2), as published by the Free Software Foundation.
  8. A copy of the license is included in the JUCE distribution, or can be found
  9. online at www.gnu.org/licenses.
  10. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  11. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  12. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  13. ------------------------------------------------------------------------------
  14. To release a closed-source product which uses JUCE, commercial licenses are
  15. available: visit www.rawmaterialsoftware.com/juce for more information.
  16. ==============================================================================
  17. */
  18. #include "../jucer_Headers.h"
  19. #include "jucer_Application.h"
  20. #include "jucer_MainWindow.h"
  21. #include "jucer_OpenDocumentManager.h"
  22. #include "../Code Editor/jucer_SourceCodeEditor.h"
  23. #include "../Project/jucer_NewProjectWizard.h"
  24. #include "../Utility/jucer_JucerTreeViewBase.h"
  25. ScopedPointer<ApplicationCommandManager> commandManager;
  26. //==============================================================================
  27. MainWindow::MainWindow()
  28. : DocumentWindow (IntrojucerApp::getApp().getApplicationName(),
  29. Colour::greyLevel (0.6f),
  30. DocumentWindow::allButtons,
  31. false)
  32. {
  33. setUsingNativeTitleBar (true);
  34. createProjectContentCompIfNeeded();
  35. #if ! JUCE_MAC
  36. setMenuBar (IntrojucerApp::getApp().menuModel);
  37. #endif
  38. setResizable (true, false);
  39. centreWithSize (800, 600);
  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. //getPeer()->setCurrentRenderingEngine (0);
  54. getLookAndFeel().setColour (ColourSelector::backgroundColourId, Colours::transparentBlack);
  55. setResizeLimits (600, 500, 32000, 32000);
  56. }
  57. MainWindow::~MainWindow()
  58. {
  59. #if ! JUCE_MAC
  60. setMenuBar (nullptr);
  61. #endif
  62. removeKeyListener (commandManager->getKeyMappings());
  63. // save the current size and position to our settings file..
  64. getGlobalProperties().setValue ("lastMainWindowPos", getWindowStateAsString());
  65. clearContentComponent();
  66. currentProject = nullptr;
  67. }
  68. void MainWindow::createProjectContentCompIfNeeded()
  69. {
  70. if (getProjectContentComponent() == nullptr)
  71. {
  72. clearContentComponent();
  73. setContentOwned (IntrojucerApp::getApp().createProjectContentComponent(), false);
  74. jassert (getProjectContentComponent() != nullptr);
  75. }
  76. }
  77. void MainWindow::makeVisible()
  78. {
  79. restoreWindowPosition();
  80. setVisible (true);
  81. addToDesktop(); // (must add before restoring size so that fullscreen will work)
  82. restoreWindowPosition();
  83. getContentComponent()->grabKeyboardFocus();
  84. }
  85. ProjectContentComponent* MainWindow::getProjectContentComponent() const
  86. {
  87. return dynamic_cast <ProjectContentComponent*> (getContentComponent());
  88. }
  89. void MainWindow::closeButtonPressed()
  90. {
  91. IntrojucerApp::getApp().mainWindowList.closeWindow (this);
  92. }
  93. bool MainWindow::closeProject (Project* project)
  94. {
  95. jassert (project == currentProject && project != nullptr);
  96. if (project == nullptr)
  97. return true;
  98. project->getStoredProperties().setValue (getProjectWindowPosName(), getWindowStateAsString());
  99. if (ProjectContentComponent* const pcc = getProjectContentComponent())
  100. {
  101. pcc->saveTreeViewState();
  102. pcc->saveOpenDocumentList();
  103. pcc->hideEditor();
  104. }
  105. if (! IntrojucerApp::getApp().openDocumentManager.closeAllDocumentsUsingProject (*project, true))
  106. return false;
  107. FileBasedDocument::SaveResult r = project->saveIfNeededAndUserAgrees();
  108. if (r == FileBasedDocument::savedOk)
  109. {
  110. setProject (nullptr);
  111. return true;
  112. }
  113. return false;
  114. }
  115. bool MainWindow::closeCurrentProject()
  116. {
  117. return currentProject == nullptr || closeProject (currentProject);
  118. }
  119. void MainWindow::setProject (Project* newProject)
  120. {
  121. createProjectContentCompIfNeeded();
  122. getProjectContentComponent()->setProject (newProject);
  123. currentProject = newProject;
  124. getProjectContentComponent()->updateMainWindowTitle();
  125. commandManager->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. || IntrojucerApp::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. if (newDoc->loadFrom (file, true)
  149. && closeCurrentProject())
  150. {
  151. setProject (newDoc.release());
  152. return true;
  153. }
  154. }
  155. else if (file.exists())
  156. {
  157. return getProjectContentComponent()->showEditorForFile (file, true);
  158. }
  159. return false;
  160. }
  161. bool MainWindow::isInterestedInFileDrag (const StringArray& filenames)
  162. {
  163. for (int i = filenames.size(); --i >= 0;)
  164. if (canOpenFile (filenames[i]))
  165. return true;
  166. return false;
  167. }
  168. void MainWindow::filesDropped (const StringArray& filenames, int /*mouseX*/, int /*mouseY*/)
  169. {
  170. for (int i = filenames.size(); --i >= 0;)
  171. {
  172. const File f (filenames[i]);
  173. if (canOpenFile (f) && openFile (f))
  174. break;
  175. }
  176. }
  177. bool MainWindow::shouldDropFilesWhenDraggedExternally (const DragAndDropTarget::SourceDetails& sourceDetails,
  178. StringArray& files, bool& canMoveFiles)
  179. {
  180. if (TreeView* tv = dynamic_cast <TreeView*> (sourceDetails.sourceComponent.get()))
  181. {
  182. Array<JucerTreeViewBase*> selected;
  183. for (int i = tv->getNumSelectedItems(); --i >= 0;)
  184. if (JucerTreeViewBase* b = dynamic_cast <JucerTreeViewBase*> (tv->getSelectedItem(i)))
  185. selected.add (b);
  186. if (selected.size() > 0)
  187. {
  188. for (int i = selected.size(); --i >= 0;)
  189. {
  190. if (JucerTreeViewBase* jtvb = selected.getUnchecked(i))
  191. {
  192. const File f (jtvb->getDraggableFile());
  193. if (f.existsAsFile())
  194. files.add (f.getFullPathName());
  195. }
  196. }
  197. canMoveFiles = false;
  198. return files.size() > 0;
  199. }
  200. }
  201. return false;
  202. }
  203. void MainWindow::activeWindowStatusChanged()
  204. {
  205. DocumentWindow::activeWindowStatusChanged();
  206. if (ProjectContentComponent* const pcc = getProjectContentComponent())
  207. pcc->updateMissingFileStatuses();
  208. IntrojucerApp::getApp().openDocumentManager.reloadModifiedFiles();
  209. }
  210. void MainWindow::updateTitle (const String& documentName)
  211. {
  212. String name (IntrojucerApp::getApp().getApplicationName());
  213. if (currentProject != nullptr)
  214. name << " - " << currentProject->getDocumentTitle();
  215. if (documentName.isNotEmpty())
  216. name << " - " << documentName;
  217. setName (name);
  218. }
  219. void MainWindow::showNewProjectWizard()
  220. {
  221. jassert (currentProject == nullptr);
  222. setContentOwned (createNewProjectWizardComponent(), true);
  223. makeVisible();
  224. }
  225. //==============================================================================
  226. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  227. {
  228. return nullptr;
  229. }
  230. void MainWindow::getAllCommands (Array <CommandID>& commands)
  231. {
  232. const CommandID ids[] = { CommandIDs::closeWindow };
  233. commands.addArray (ids, numElementsInArray (ids));
  234. }
  235. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  236. {
  237. switch (commandID)
  238. {
  239. case CommandIDs::closeWindow:
  240. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  241. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  242. break;
  243. default:
  244. break;
  245. }
  246. }
  247. bool MainWindow::perform (const InvocationInfo& info)
  248. {
  249. switch (info.commandID)
  250. {
  251. case CommandIDs::closeWindow:
  252. closeButtonPressed();
  253. break;
  254. default:
  255. return false;
  256. }
  257. return true;
  258. }
  259. //==============================================================================
  260. MainWindowList::MainWindowList()
  261. {
  262. }
  263. void MainWindowList::forceCloseAllWindows()
  264. {
  265. windows.clear();
  266. }
  267. bool MainWindowList::askAllWindowsToClose()
  268. {
  269. saveCurrentlyOpenProjectList();
  270. while (windows.size() > 0)
  271. {
  272. if (! windows[0]->closeCurrentProject())
  273. return false;
  274. windows.remove (0);
  275. }
  276. return true;
  277. }
  278. void MainWindowList::createWindowIfNoneAreOpen()
  279. {
  280. if (windows.size() == 0)
  281. createNewMainWindow()->makeVisible();
  282. }
  283. void MainWindowList::closeWindow (MainWindow* w)
  284. {
  285. jassert (windows.contains (w));
  286. #if ! JUCE_MAC
  287. if (windows.size() == 1)
  288. {
  289. JUCEApplication::getInstance()->systemRequestedQuit();
  290. }
  291. else
  292. #endif
  293. {
  294. if (w->closeCurrentProject())
  295. {
  296. windows.removeObject (w);
  297. saveCurrentlyOpenProjectList();
  298. }
  299. }
  300. }
  301. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  302. {
  303. MainWindow* w = getOrCreateFrontmostWindow();
  304. w->getProjectContentComponent()->showDocument (doc, grabFocus);
  305. }
  306. bool MainWindowList::openFile (const File& file)
  307. {
  308. for (int i = windows.size(); --i >= 0;)
  309. {
  310. MainWindow* const w = windows.getUnchecked(i);
  311. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  312. {
  313. w->toFront (true);
  314. return true;
  315. }
  316. }
  317. if (file.hasFileExtension (Project::projectFileExtension))
  318. {
  319. ScopedPointer <Project> newDoc (new Project (file));
  320. if (newDoc->loadFrom (file, true))
  321. {
  322. MainWindow* const w = getOrCreateEmptyWindow();
  323. w->setProject (newDoc);
  324. newDoc.release()->setChangedFlag (false);
  325. w->makeVisible();
  326. avoidSuperimposedWindows (w);
  327. jassert (w->getProjectContentComponent() != nullptr);
  328. w->getProjectContentComponent()->reloadLastOpenDocuments();
  329. return true;
  330. }
  331. }
  332. else if (file.exists())
  333. {
  334. MainWindow* const w = getOrCreateFrontmostWindow();
  335. return w->openFile (file);
  336. }
  337. return false;
  338. }
  339. MainWindow* MainWindowList::createNewMainWindow()
  340. {
  341. MainWindow* const w = new MainWindow();
  342. windows.add (w);
  343. w->restoreWindowPosition();
  344. avoidSuperimposedWindows (w);
  345. return w;
  346. }
  347. MainWindow* MainWindowList::getOrCreateFrontmostWindow()
  348. {
  349. if (windows.size() == 0)
  350. {
  351. MainWindow* w = createNewMainWindow();
  352. avoidSuperimposedWindows (w);
  353. w->makeVisible();
  354. return w;
  355. }
  356. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  357. {
  358. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  359. if (windows.contains (mw))
  360. return mw;
  361. }
  362. return windows.getLast();
  363. }
  364. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  365. {
  366. if (windows.size() == 0)
  367. return createNewMainWindow();
  368. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  369. {
  370. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  371. if (windows.contains (mw) && mw->getProject() == nullptr)
  372. return mw;
  373. }
  374. return createNewMainWindow();
  375. }
  376. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  377. {
  378. for (int i = windows.size(); --i >= 0;)
  379. {
  380. MainWindow* const other = windows.getUnchecked(i);
  381. const Rectangle<int> b1 (mw->getBounds());
  382. const Rectangle<int> b2 (other->getBounds());
  383. if (mw != other
  384. && std::abs (b1.getX() - b2.getX()) < 3
  385. && std::abs (b1.getY() - b2.getY()) < 3
  386. && std::abs (b1.getRight() - b2.getRight()) < 3
  387. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  388. {
  389. int dx = 40, dy = 30;
  390. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  391. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  392. mw->setBounds (b1.translated (dx, dy));
  393. }
  394. }
  395. }
  396. void MainWindowList::saveCurrentlyOpenProjectList()
  397. {
  398. Array<File> projects;
  399. Desktop& desktop = Desktop::getInstance();
  400. for (int i = 0; i < desktop.getNumComponents(); ++i)
  401. {
  402. if (MainWindow* const mw = dynamic_cast <MainWindow*> (desktop.getComponent(i)))
  403. if (Project* p = mw->getProject())
  404. projects.add (p->getFile());
  405. }
  406. getAppSettings().setLastProjects (projects);
  407. }
  408. void MainWindowList::reopenLastProjects()
  409. {
  410. Array<File> projects (getAppSettings().getLastProjects());
  411. for (int i = 0; i < projects.size(); ++ i)
  412. openFile (projects.getReference(i));
  413. }
  414. void MainWindowList::sendLookAndFeelChange()
  415. {
  416. for (int i = windows.size(); --i >= 0;)
  417. windows.getUnchecked(i)->sendLookAndFeelChange();
  418. }