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.

531 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. ProjectContentComponent* const pcc = getProjectContentComponent();
  100. if (pcc != nullptr)
  101. {
  102. pcc->saveTreeViewState();
  103. pcc->saveOpenDocumentList();
  104. pcc->hideEditor();
  105. }
  106. if (! IntrojucerApp::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. getProjectContentComponent()->updateMainWindowTitle();
  126. commandManager->commandStatusChanged();
  127. }
  128. void MainWindow::restoreWindowPosition()
  129. {
  130. String windowState;
  131. if (currentProject != nullptr)
  132. windowState = currentProject->getStoredProperties().getValue (getProjectWindowPosName());
  133. if (windowState.isEmpty())
  134. windowState = getGlobalProperties().getValue ("lastMainWindowPos");
  135. restoreWindowStateFromString (windowState);
  136. }
  137. bool MainWindow::canOpenFile (const File& file) const
  138. {
  139. return 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. const File f (selected.getUnchecked(i)->getDraggableFile());
  191. if (f.existsAsFile())
  192. files.add (f.getFullPathName());
  193. }
  194. canMoveFiles = false;
  195. return files.size() > 0;
  196. }
  197. }
  198. return false;
  199. }
  200. void MainWindow::activeWindowStatusChanged()
  201. {
  202. DocumentWindow::activeWindowStatusChanged();
  203. if (getProjectContentComponent() != nullptr)
  204. getProjectContentComponent()->updateMissingFileStatuses();
  205. IntrojucerApp::getApp().openDocumentManager.reloadModifiedFiles();
  206. }
  207. void MainWindow::updateTitle (const String& documentName)
  208. {
  209. String name (IntrojucerApp::getApp().getApplicationName());
  210. if (currentProject != nullptr)
  211. name << " - " << currentProject->getDocumentTitle();
  212. if (documentName.isNotEmpty())
  213. name << " - " << documentName;
  214. setName (name);
  215. }
  216. void MainWindow::showNewProjectWizard()
  217. {
  218. jassert (currentProject == nullptr);
  219. setContentOwned (NewProjectWizard::createComponent(), true);
  220. makeVisible();
  221. }
  222. //==============================================================================
  223. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  224. {
  225. return nullptr;
  226. }
  227. void MainWindow::getAllCommands (Array <CommandID>& commands)
  228. {
  229. const CommandID ids[] = { CommandIDs::closeWindow };
  230. commands.addArray (ids, numElementsInArray (ids));
  231. }
  232. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  233. {
  234. switch (commandID)
  235. {
  236. case CommandIDs::closeWindow:
  237. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  238. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  239. break;
  240. default:
  241. break;
  242. }
  243. }
  244. bool MainWindow::perform (const InvocationInfo& info)
  245. {
  246. switch (info.commandID)
  247. {
  248. case CommandIDs::closeWindow:
  249. closeButtonPressed();
  250. break;
  251. default:
  252. return false;
  253. }
  254. return true;
  255. }
  256. //==============================================================================
  257. MainWindowList::MainWindowList()
  258. {
  259. }
  260. void MainWindowList::forceCloseAllWindows()
  261. {
  262. windows.clear();
  263. }
  264. bool MainWindowList::askAllWindowsToClose()
  265. {
  266. saveCurrentlyOpenProjectList();
  267. while (windows.size() > 0)
  268. {
  269. if (! windows[0]->closeCurrentProject())
  270. return false;
  271. windows.remove (0);
  272. }
  273. return true;
  274. }
  275. void MainWindowList::createWindowIfNoneAreOpen()
  276. {
  277. if (windows.size() == 0)
  278. createNewMainWindow()->makeVisible();
  279. }
  280. void MainWindowList::closeWindow (MainWindow* w)
  281. {
  282. jassert (windows.contains (w));
  283. #if ! JUCE_MAC
  284. if (windows.size() == 1)
  285. {
  286. JUCEApplication::getInstance()->systemRequestedQuit();
  287. }
  288. else
  289. #endif
  290. {
  291. if (w->closeCurrentProject())
  292. {
  293. windows.removeObject (w);
  294. saveCurrentlyOpenProjectList();
  295. }
  296. }
  297. }
  298. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  299. {
  300. MainWindow* w = getOrCreateFrontmostWindow();
  301. w->getProjectContentComponent()->showDocument (doc, grabFocus);
  302. }
  303. bool MainWindowList::openFile (const File& file)
  304. {
  305. for (int i = windows.size(); --i >= 0;)
  306. {
  307. MainWindow* const w = windows.getUnchecked(i);
  308. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  309. {
  310. w->toFront (true);
  311. return true;
  312. }
  313. }
  314. if (file.hasFileExtension (Project::projectFileExtension))
  315. {
  316. ScopedPointer <Project> newDoc (new Project (file));
  317. if (newDoc->loadFrom (file, true))
  318. {
  319. MainWindow* const w = getOrCreateEmptyWindow();
  320. w->setProject (newDoc);
  321. newDoc.release()->setChangedFlag (false);
  322. w->makeVisible();
  323. avoidSuperimposedWindows (w);
  324. jassert (w->getProjectContentComponent() != nullptr);
  325. w->getProjectContentComponent()->reloadLastOpenDocuments();
  326. return true;
  327. }
  328. }
  329. else if (file.exists())
  330. {
  331. MainWindow* const w = getOrCreateFrontmostWindow();
  332. return w->openFile (file);
  333. }
  334. return false;
  335. }
  336. MainWindow* MainWindowList::createNewMainWindow()
  337. {
  338. MainWindow* const w = new MainWindow();
  339. windows.add (w);
  340. w->restoreWindowPosition();
  341. avoidSuperimposedWindows (w);
  342. return w;
  343. }
  344. MainWindow* MainWindowList::getOrCreateFrontmostWindow()
  345. {
  346. if (windows.size() == 0)
  347. {
  348. MainWindow* w = createNewMainWindow();
  349. avoidSuperimposedWindows (w);
  350. w->makeVisible();
  351. return w;
  352. }
  353. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  354. {
  355. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  356. if (windows.contains (mw))
  357. return mw;
  358. }
  359. return windows.getLast();
  360. }
  361. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  362. {
  363. if (windows.size() == 0)
  364. return createNewMainWindow();
  365. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  366. {
  367. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  368. if (windows.contains (mw) && mw->getProject() == nullptr)
  369. return mw;
  370. }
  371. return createNewMainWindow();
  372. }
  373. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  374. {
  375. for (int i = windows.size(); --i >= 0;)
  376. {
  377. MainWindow* const other = windows.getUnchecked(i);
  378. const Rectangle<int> b1 (mw->getBounds());
  379. const Rectangle<int> b2 (other->getBounds());
  380. if (mw != other
  381. && std::abs (b1.getX() - b2.getX()) < 3
  382. && std::abs (b1.getY() - b2.getY()) < 3
  383. && std::abs (b1.getRight() - b2.getRight()) < 3
  384. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  385. {
  386. int dx = 40, dy = 30;
  387. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  388. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  389. mw->setBounds (b1.translated (dx, dy));
  390. }
  391. }
  392. }
  393. void MainWindowList::saveCurrentlyOpenProjectList()
  394. {
  395. Array<File> projects;
  396. Desktop& desktop = Desktop::getInstance();
  397. for (int i = 0; i < desktop.getNumComponents(); ++i)
  398. {
  399. MainWindow* const mw = dynamic_cast <MainWindow*> (desktop.getComponent(i));
  400. if (mw != nullptr && mw->getProject() != nullptr)
  401. projects.add (mw->getProject()->getFile());
  402. }
  403. getAppSettings().setLastProjects (projects);
  404. }
  405. void MainWindowList::reopenLastProjects()
  406. {
  407. Array<File> projects (getAppSettings().getLastProjects());
  408. for (int i = 0; i < projects.size(); ++ i)
  409. openFile (projects.getReference(i));
  410. }
  411. void MainWindowList::sendLookAndFeelChange()
  412. {
  413. for (int i = windows.size(); --i >= 0;)
  414. windows.getUnchecked(i)->sendLookAndFeelChange();
  415. }