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.

483 lines
13KB

  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 (JucerApplication::getApp()->getApplicationName(),
  28. Colour::greyLevel (0.6f),
  29. DocumentWindow::allButtons,
  30. false)
  31. {
  32. setUsingNativeTitleBar (true);
  33. createProjectContentCompIfNeeded();
  34. #if ! JUCE_MAC
  35. setMenuBar (JucerApplication::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 (StoredSettings::getInstance()->getProps().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. StoredSettings::getInstance()->getProps()
  63. .setValue ("lastMainWindowPos", getWindowStateAsString());
  64. clearContentComponent();
  65. currentProject = nullptr;
  66. }
  67. void MainWindow::createProjectContentCompIfNeeded()
  68. {
  69. if (getProjectContentComponent() == nullptr)
  70. {
  71. clearContentComponent();
  72. setContentOwned (JucerApplication::getApp()->createProjectContentComponent(), false);
  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. JucerApplication::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. StoredSettings::getInstance()->getProps()
  97. .setValue (getProjectWindowPosName(), getWindowStateAsString());
  98. if (! OpenDocumentManager::getInstance()->closeAllDocumentsUsingProject (*project, true))
  99. return false;
  100. ProjectContentComponent* const pcc = getProjectContentComponent();
  101. if (pcc != nullptr)
  102. pcc->saveTreeViewState();
  103. FileBasedDocument::SaveResult r = project->saveIfNeededAndUserAgrees();
  104. if (r == FileBasedDocument::savedOk)
  105. {
  106. setProject (nullptr);
  107. return true;
  108. }
  109. return false;
  110. }
  111. bool MainWindow::closeCurrentProject()
  112. {
  113. return currentProject == nullptr || closeProject (currentProject);
  114. }
  115. void MainWindow::setProject (Project* newProject)
  116. {
  117. createProjectContentCompIfNeeded();
  118. getProjectContentComponent()->setProject (newProject);
  119. currentProject = newProject;
  120. commandManager->commandStatusChanged();
  121. }
  122. void MainWindow::restoreWindowPosition()
  123. {
  124. String windowState;
  125. if (currentProject != nullptr)
  126. windowState = StoredSettings::getInstance()->getProps().getValue (getProjectWindowPosName());
  127. if (windowState.isEmpty())
  128. windowState = StoredSettings::getInstance()->getProps().getValue ("lastMainWindowPos");
  129. restoreWindowStateFromString (windowState);
  130. }
  131. bool MainWindow::canOpenFile (const File& file) const
  132. {
  133. return file.hasFileExtension (Project::projectFileExtension)
  134. || OpenDocumentManager::getInstance()->canOpenFile (file);
  135. }
  136. bool MainWindow::openFile (const File& file)
  137. {
  138. createProjectContentCompIfNeeded();
  139. if (file.hasFileExtension (Project::projectFileExtension))
  140. {
  141. ScopedPointer <Project> newDoc (new Project (file));
  142. if (newDoc->loadFrom (file, true)
  143. && closeCurrentProject())
  144. {
  145. setProject (newDoc.release());
  146. return true;
  147. }
  148. }
  149. else if (file.exists())
  150. {
  151. return getProjectContentComponent()->showEditorForFile (file);
  152. }
  153. return false;
  154. }
  155. bool MainWindow::isInterestedInFileDrag (const StringArray& filenames)
  156. {
  157. for (int i = filenames.size(); --i >= 0;)
  158. if (canOpenFile (filenames[i]))
  159. return true;
  160. return false;
  161. }
  162. void MainWindow::filesDropped (const StringArray& filenames, int mouseX, int mouseY)
  163. {
  164. for (int i = filenames.size(); --i >= 0;)
  165. {
  166. const File f (filenames[i]);
  167. if (canOpenFile (f) && openFile (f))
  168. break;
  169. }
  170. }
  171. void MainWindow::activeWindowStatusChanged()
  172. {
  173. DocumentWindow::activeWindowStatusChanged();
  174. if (getProjectContentComponent() != nullptr)
  175. getProjectContentComponent()->updateMissingFileStatuses();
  176. OpenDocumentManager::getInstance()->reloadModifiedFiles();
  177. }
  178. void MainWindow::updateTitle (const String& documentName)
  179. {
  180. String name (JucerApplication::getApp()->getApplicationName());
  181. if (currentProject != nullptr)
  182. name = currentProject->getDocumentTitle() + " - " + name;
  183. if (documentName.isNotEmpty())
  184. name = documentName + " - " + name;
  185. setName (name);
  186. }
  187. void MainWindow::showNewProjectWizard()
  188. {
  189. jassert (currentProject == nullptr);
  190. setContentOwned (NewProjectWizard::createComponent(), true);
  191. makeVisible();
  192. }
  193. //==============================================================================
  194. ApplicationCommandTarget* MainWindow::getNextCommandTarget()
  195. {
  196. return 0;
  197. }
  198. void MainWindow::getAllCommands (Array <CommandID>& commands)
  199. {
  200. const CommandID ids[] = { CommandIDs::closeWindow };
  201. commands.addArray (ids, numElementsInArray (ids));
  202. }
  203. void MainWindow::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  204. {
  205. switch (commandID)
  206. {
  207. case CommandIDs::closeWindow:
  208. result.setInfo ("Close Window", "Closes the current window", CommandCategories::general, 0);
  209. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier, 0));
  210. break;
  211. default:
  212. break;
  213. }
  214. }
  215. bool MainWindow::perform (const InvocationInfo& info)
  216. {
  217. switch (info.commandID)
  218. {
  219. case CommandIDs::closeWindow:
  220. closeButtonPressed();
  221. break;
  222. default:
  223. return false;
  224. }
  225. return true;
  226. }
  227. //==============================================================================
  228. MainWindowList::MainWindowList()
  229. {
  230. }
  231. void MainWindowList::forceCloseAllWindows()
  232. {
  233. windows.clear();
  234. }
  235. bool MainWindowList::askAllWindowsToClose()
  236. {
  237. saveCurrentlyOpenProjectList();
  238. while (windows.size() > 0)
  239. {
  240. if (! windows[0]->closeCurrentProject())
  241. return false;
  242. windows.remove (0);
  243. }
  244. return true;
  245. }
  246. void MainWindowList::createWindowIfNoneAreOpen()
  247. {
  248. if (windows.size() == 0)
  249. createNewMainWindow()->makeVisible();
  250. }
  251. void MainWindowList::closeWindow (MainWindow* w)
  252. {
  253. jassert (windows.contains (w));
  254. #if ! JUCE_MAC
  255. if (windows.size() == 1)
  256. {
  257. JUCEApplication::getInstance()->systemRequestedQuit();
  258. }
  259. else
  260. #endif
  261. {
  262. if (w->closeCurrentProject())
  263. {
  264. windows.removeObject (w);
  265. saveCurrentlyOpenProjectList();
  266. }
  267. }
  268. }
  269. void MainWindowList::openDocument (OpenDocumentManager::Document* doc)
  270. {
  271. MainWindow* const w = getOrCreateFrontmostWindow();
  272. w->makeVisible();
  273. w->getProjectContentComponent()->showDocument (doc);
  274. avoidSuperimposedWindows (w);
  275. }
  276. bool MainWindowList::openFile (const File& file)
  277. {
  278. for (int i = windows.size(); --i >= 0;)
  279. {
  280. MainWindow* const w = windows.getUnchecked(i);
  281. if (w->getProject() != nullptr && w->getProject()->getFile() == file)
  282. {
  283. w->toFront (true);
  284. return true;
  285. }
  286. }
  287. if (file.hasFileExtension (Project::projectFileExtension))
  288. {
  289. ScopedPointer <Project> newDoc (new Project (file));
  290. if (newDoc->loadFrom (file, true))
  291. {
  292. MainWindow* const w = getOrCreateEmptyWindow();
  293. w->setProject (newDoc.release());
  294. w->makeVisible();
  295. avoidSuperimposedWindows (w);
  296. return true;
  297. }
  298. }
  299. else if (file.exists())
  300. {
  301. MainWindow* const w = getOrCreateFrontmostWindow();
  302. const bool ok = w->openFile (file);
  303. w->makeVisible();
  304. avoidSuperimposedWindows (w);
  305. return ok;
  306. }
  307. return false;
  308. }
  309. MainWindow* MainWindowList::createNewMainWindow()
  310. {
  311. MainWindow* const w = new MainWindow();
  312. windows.add (w);
  313. w->restoreWindowPosition();
  314. avoidSuperimposedWindows (w);
  315. return w;
  316. }
  317. MainWindow* MainWindowList::getOrCreateFrontmostWindow()
  318. {
  319. if (windows.size() == 0)
  320. return createNewMainWindow();
  321. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  322. {
  323. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  324. if (windows.contains (mw))
  325. return mw;
  326. }
  327. return windows.getLast();
  328. }
  329. MainWindow* MainWindowList::getOrCreateEmptyWindow()
  330. {
  331. if (windows.size() == 0)
  332. return createNewMainWindow();
  333. for (int i = Desktop::getInstance().getNumComponents(); --i >= 0;)
  334. {
  335. MainWindow* mw = dynamic_cast <MainWindow*> (Desktop::getInstance().getComponent (i));
  336. if (windows.contains (mw) && mw->getProject() == nullptr)
  337. return mw;
  338. }
  339. return createNewMainWindow();
  340. }
  341. void MainWindowList::avoidSuperimposedWindows (MainWindow* const mw)
  342. {
  343. for (int i = windows.size(); --i >= 0;)
  344. {
  345. MainWindow* const other = windows.getUnchecked(i);
  346. const Rectangle<int> b1 (mw->getBounds());
  347. const Rectangle<int> b2 (other->getBounds());
  348. if (mw != other
  349. && std::abs (b1.getX() - b2.getX()) < 3
  350. && std::abs (b1.getY() - b2.getY()) < 3
  351. && std::abs (b1.getRight() - b2.getRight()) < 3
  352. && std::abs (b1.getBottom() - b2.getBottom()) < 3)
  353. {
  354. int dx = 40, dy = 30;
  355. if (b1.getCentreX() >= mw->getScreenBounds().getCentreX()) dx = -dx;
  356. if (b1.getCentreY() >= mw->getScreenBounds().getCentreY()) dy = -dy;
  357. mw->setBounds (b1.translated (dx, dy));
  358. }
  359. }
  360. }
  361. void MainWindowList::saveCurrentlyOpenProjectList()
  362. {
  363. Array<File> projects;
  364. Desktop& desktop = Desktop::getInstance();
  365. for (int i = 0; i < desktop.getNumComponents(); ++i)
  366. {
  367. MainWindow* const mw = dynamic_cast <MainWindow*> (desktop.getComponent(i));
  368. if (mw != nullptr && mw->getProject() != nullptr)
  369. projects.add (mw->getProject()->getFile());
  370. }
  371. StoredSettings::getInstance()->setLastProjects (projects);
  372. }
  373. void MainWindowList::reopenLastProjects()
  374. {
  375. Array<File> projects (StoredSettings::getInstance()->getLastProjects());
  376. for (int i = 0; i < projects.size(); ++ i)
  377. openFile (projects.getReference(i));
  378. }