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.

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