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.

499 lines
14KB

  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. ScopedPointer<ApplicationCommandManager> commandManager;
  25. //==============================================================================
  26. MainWindow::MainWindow()
  27. : DocumentWindow (IntrojucerApp::getApp().getApplicationName(),
  28. Colour::greyLevel (0.6f),
  29. DocumentWindow::allButtons,
  30. false)
  31. {
  32. setUsingNativeTitleBar (true);
  33. createProjectContentCompIfNeeded();
  34. #if ! JUCE_MAC
  35. setMenuBar (IntrojucerApp::getApp().menuModel);
  36. #endif
  37. setResizable (true, false);
  38. centreWithSize (800, 600);
  39. // Register all the app commands..
  40. commandManager->registerAllCommandsForTarget (this);
  41. commandManager->registerAllCommandsForTarget (getProjectContentComponent());
  42. // update key mappings..
  43. {
  44. commandManager->getKeyMappings()->resetToDefaultMappings();
  45. ScopedPointer <XmlElement> keys (getGlobalProperties().getXmlValue ("keyMappings"));
  46. if (keys != nullptr)
  47. commandManager->getKeyMappings()->restoreFromXml (*keys);
  48. addKeyListener (commandManager->getKeyMappings());
  49. }
  50. // don't want the window to take focus when the title-bar is clicked..
  51. setWantsKeyboardFocus (false);
  52. //getPeer()->setCurrentRenderingEngine (0);
  53. getLookAndFeel().setColour (ColourSelector::backgroundColourId, Colours::transparentBlack);
  54. }
  55. MainWindow::~MainWindow()
  56. {
  57. #if ! JUCE_MAC
  58. setMenuBar (nullptr);
  59. #endif
  60. removeKeyListener (commandManager->getKeyMappings());
  61. // save the current size and position to our settings file..
  62. getGlobalProperties().setValue ("lastMainWindowPos", getWindowStateAsString());
  63. clearContentComponent();
  64. currentProject = nullptr;
  65. }
  66. void MainWindow::createProjectContentCompIfNeeded()
  67. {
  68. if (getProjectContentComponent() == nullptr)
  69. {
  70. clearContentComponent();
  71. setContentOwned (IntrojucerApp::getApp().createProjectContentComponent(), false);
  72. jassert (getProjectContentComponent() != nullptr);
  73. }
  74. }
  75. void MainWindow::makeVisible()
  76. {
  77. restoreWindowPosition();
  78. setVisible (true);
  79. addToDesktop(); // (must add before restoring size so that fullscreen will work)
  80. restoreWindowPosition();
  81. getContentComponent()->grabKeyboardFocus();
  82. }
  83. ProjectContentComponent* MainWindow::getProjectContentComponent() const
  84. {
  85. return dynamic_cast <ProjectContentComponent*> (getContentComponent());
  86. }
  87. void MainWindow::closeButtonPressed()
  88. {
  89. IntrojucerApp::getApp().mainWindowList.closeWindow (this);
  90. }
  91. bool MainWindow::closeProject (Project* project)
  92. {
  93. jassert (project == currentProject && project != nullptr);
  94. if (project == nullptr)
  95. return true;
  96. project->getStoredProperties().setValue (getProjectWindowPosName(), getWindowStateAsString());
  97. ProjectContentComponent* const pcc = getProjectContentComponent();
  98. if (pcc != nullptr)
  99. {
  100. pcc->saveTreeViewState();
  101. pcc->saveOpenDocumentList();
  102. pcc->hideEditor();
  103. }
  104. if (! IntrojucerApp::getApp().openDocumentManager.closeAllDocumentsUsingProject (*project, true))
  105. return false;
  106. FileBasedDocument::SaveResult r = project->saveIfNeededAndUserAgrees();
  107. if (r == FileBasedDocument::savedOk)
  108. {
  109. setProject (nullptr);
  110. return true;
  111. }
  112. return false;
  113. }
  114. bool MainWindow::closeCurrentProject()
  115. {
  116. return currentProject == nullptr || closeProject (currentProject);
  117. }
  118. void MainWindow::setProject (Project* newProject)
  119. {
  120. createProjectContentCompIfNeeded();
  121. getProjectContentComponent()->setProject (newProject);
  122. currentProject = newProject;
  123. getProjectContentComponent()->updateMainWindowTitle();
  124. commandManager->commandStatusChanged();
  125. }
  126. void MainWindow::restoreWindowPosition()
  127. {
  128. String windowState;
  129. if (currentProject != nullptr)
  130. windowState = currentProject->getStoredProperties().getValue (getProjectWindowPosName());
  131. if (windowState.isEmpty())
  132. windowState = getGlobalProperties().getValue ("lastMainWindowPos");
  133. restoreWindowStateFromString (windowState);
  134. }
  135. bool MainWindow::canOpenFile (const File& file) const
  136. {
  137. return file.hasFileExtension (Project::projectFileExtension)
  138. || IntrojucerApp::getApp().openDocumentManager.canOpenFile (file);
  139. }
  140. bool MainWindow::openFile (const File& file)
  141. {
  142. createProjectContentCompIfNeeded();
  143. if (file.hasFileExtension (Project::projectFileExtension))
  144. {
  145. ScopedPointer <Project> newDoc (new Project (file));
  146. if (newDoc->loadFrom (file, true)
  147. && closeCurrentProject())
  148. {
  149. setProject (newDoc.release());
  150. return true;
  151. }
  152. }
  153. else if (file.exists())
  154. {
  155. return getProjectContentComponent()->showEditorForFile (file, true);
  156. }
  157. return false;
  158. }
  159. bool MainWindow::isInterestedInFileDrag (const StringArray& filenames)
  160. {
  161. for (int i = filenames.size(); --i >= 0;)
  162. if (canOpenFile (filenames[i]))
  163. return true;
  164. return false;
  165. }
  166. void MainWindow::filesDropped (const StringArray& filenames, int mouseX, int mouseY)
  167. {
  168. for (int i = filenames.size(); --i >= 0;)
  169. {
  170. const File f (filenames[i]);
  171. if (canOpenFile (f) && openFile (f))
  172. break;
  173. }
  174. }
  175. void MainWindow::activeWindowStatusChanged()
  176. {
  177. DocumentWindow::activeWindowStatusChanged();
  178. if (getProjectContentComponent() != nullptr)
  179. getProjectContentComponent()->updateMissingFileStatuses();
  180. IntrojucerApp::getApp().openDocumentManager.reloadModifiedFiles();
  181. }
  182. void MainWindow::updateTitle (const String& documentName)
  183. {
  184. String name (IntrojucerApp::getApp().getApplicationName());
  185. if (currentProject != nullptr)
  186. name << " - " << currentProject->getDocumentTitle();
  187. if (documentName.isNotEmpty())
  188. name << " - " << documentName;
  189. setName (name);
  190. }
  191. void MainWindow::showNewProjectWizard()
  192. {
  193. jassert (currentProject == nullptr);
  194. setContentOwned (NewProjectWizard::createComponent(), true);
  195. makeVisible();
  196. }
  197. //==============================================================================
  198. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  199. {
  200. return 0;
  201. }
  202. void MainWindow::getAllCommands (Array <CommandID>& commands)
  203. {
  204. const CommandID ids[] = { CommandIDs::closeWindow };
  205. commands.addArray (ids, numElementsInArray (ids));
  206. }
  207. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  208. {
  209. switch (commandID)
  210. {
  211. case CommandIDs::closeWindow:
  212. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  213. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  214. break;
  215. default:
  216. break;
  217. }
  218. }
  219. bool MainWindow::perform (const InvocationInfo& info)
  220. {
  221. switch (info.commandID)
  222. {
  223. case CommandIDs::closeWindow:
  224. closeButtonPressed();
  225. break;
  226. default:
  227. return false;
  228. }
  229. return true;
  230. }
  231. //==============================================================================
  232. MainWindowList::MainWindowList()
  233. {
  234. }
  235. void MainWindowList::forceCloseAllWindows()
  236. {
  237. windows.clear();
  238. }
  239. bool MainWindowList::askAllWindowsToClose()
  240. {
  241. saveCurrentlyOpenProjectList();
  242. while (windows.size() > 0)
  243. {
  244. if (! windows[0]->closeCurrentProject())
  245. return false;
  246. windows.remove (0);
  247. }
  248. return true;
  249. }
  250. void MainWindowList::createWindowIfNoneAreOpen()
  251. {
  252. if (windows.size() == 0)
  253. createNewMainWindow()->makeVisible();
  254. }
  255. void MainWindowList::closeWindow (MainWindow* w)
  256. {
  257. jassert (windows.contains (w));
  258. #if ! JUCE_MAC
  259. if (windows.size() == 1)
  260. {
  261. JUCEApplication::getInstance()->systemRequestedQuit();
  262. }
  263. else
  264. #endif
  265. {
  266. if (w->closeCurrentProject())
  267. {
  268. windows.removeObject (w);
  269. saveCurrentlyOpenProjectList();
  270. }
  271. }
  272. }
  273. void MainWindowList::openDocument (OpenDocumentManager::Document* doc, bool grabFocus)
  274. {
  275. MainWindow* w = getOrCreateFrontmostWindow();
  276. w->getProjectContentComponent()->showDocument (doc, grabFocus);
  277. }
  278. bool MainWindowList::openFile (const File& file)
  279. {
  280. for (int i = windows.size(); --i >= 0;)
  281. {
  282. MainWindow* const w = windows.getUnchecked(i);
  283. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  284. {
  285. w->toFront (true);
  286. return true;
  287. }
  288. }
  289. if (file.hasFileExtension (Project::projectFileExtension))
  290. {
  291. ScopedPointer <Project> newDoc (new Project (file));
  292. if (newDoc->loadFrom (file, true))
  293. {
  294. MainWindow* const w = getOrCreateEmptyWindow();
  295. w->setProject (newDoc);
  296. newDoc.release()->setChangedFlag (false);
  297. w->makeVisible();
  298. avoidSuperimposedWindows (w);
  299. jassert (w->getProjectContentComponent() != nullptr);
  300. w->getProjectContentComponent()->reloadLastOpenDocuments();
  301. return true;
  302. }
  303. }
  304. else if (file.exists())
  305. {
  306. MainWindow* const w = getOrCreateFrontmostWindow();
  307. return w->openFile (file);
  308. }
  309. return false;
  310. }
  311. MainWindow* MainWindowList::createNewMainWindow()
  312. {
  313. MainWindow* const w = new MainWindow();
  314. windows.add (w);
  315. w->restoreWindowPosition();
  316. avoidSuperimposedWindows (w);
  317. return w;
  318. }
  319. MainWindow* MainWindowList::getOrCreateFrontmostWindow()
  320. {
  321. if (windows.size() == 0)
  322. {
  323. MainWindow* w = createNewMainWindow();
  324. avoidSuperimposedWindows (w);
  325. w->makeVisible();
  326. return w;
  327. }
  328. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  329. {
  330. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  331. if (windows.contains (mw))
  332. return mw;
  333. }
  334. return windows.getLast();
  335. }
  336. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  337. {
  338. if (windows.size() == 0)
  339. return createNewMainWindow();
  340. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  341. {
  342. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  343. if (windows.contains (mw) && mw->getProject() == nullptr)
  344. return mw;
  345. }
  346. return createNewMainWindow();
  347. }
  348. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  349. {
  350. for (int i = windows.size(); --i >= 0;)
  351. {
  352. MainWindow* const other = windows.getUnchecked(i);
  353. const Rectangle<int> b1 (mw->getBounds());
  354. const Rectangle<int> b2 (other->getBounds());
  355. if (mw != other
  356. && std::abs (b1.getX() - b2.getX()) < 3
  357. && std::abs (b1.getY() - b2.getY()) < 3
  358. && std::abs (b1.getRight() - b2.getRight()) < 3
  359. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  360. {
  361. int dx = 40, dy = 30;
  362. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  363. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  364. mw->setBounds (b1.translated (dx, dy));
  365. }
  366. }
  367. }
  368. void MainWindowList::saveCurrentlyOpenProjectList()
  369. {
  370. Array<File> projects;
  371. Desktop& desktop = Desktop::getInstance();
  372. for (int i = 0; i < desktop.getNumComponents(); ++i)
  373. {
  374. MainWindow* const mw = dynamic_cast <MainWindow*> (desktop.getComponent(i));
  375. if (mw != nullptr && mw->getProject() != nullptr)
  376. projects.add (mw->getProject()->getFile());
  377. }
  378. getAppSettings().setLastProjects (projects);
  379. }
  380. void MainWindowList::reopenLastProjects()
  381. {
  382. Array<File> projects (getAppSettings().getLastProjects());
  383. for (int i = 0; i < projects.size(); ++ i)
  384. openFile (projects.getReference(i));
  385. }
  386. void MainWindowList::sendLookAndFeelChange()
  387. {
  388. for (int i = windows.size(); --i >= 0;)
  389. windows.getUnchecked(i)->sendLookAndFeelChange();
  390. }