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.

663 lines
22KB

  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_ProjectContentComponent.h"
  19. #include "../Application/jucer_MainWindow.h"
  20. #include "../Application/jucer_Application.h"
  21. #include "../Code Editor/jucer_SourceCodeEditor.h"
  22. #include "jucer_ConfigPage.h"
  23. #include "jucer_TreeViewTypes.h"
  24. #include "../Project Saving/jucer_ProjectExporter.h"
  25. //==============================================================================
  26. class FileTreeTab : public TreePanelBase
  27. {
  28. public:
  29. FileTreeTab (Project& project)
  30. : TreePanelBase ("treeViewState_" + project.getProjectUID())
  31. {
  32. tree.setMultiSelectEnabled (true);
  33. setRoot (new GroupTreeViewItem (project.getMainGroup()));
  34. }
  35. };
  36. //==============================================================================
  37. class ConfigTreeTab : public TreePanelBase
  38. {
  39. public:
  40. ConfigTreeTab (Project& project)
  41. : TreePanelBase ("settingsTreeViewState_" + project.getProjectUID())
  42. {
  43. tree.setMultiSelectEnabled (false);
  44. setRoot (createProjectConfigTreeViewRoot (project));
  45. if (tree.getNumSelectedItems() == 0)
  46. tree.getRootItem()->setSelected (true, true);
  47. #if JUCE_MAC || JUCE_WINDOWS
  48. addAndMakeVisible (&openProjectButton);
  49. openProjectButton.setCommandToTrigger (commandManager, CommandIDs::openInIDE, true);
  50. openProjectButton.setButtonText (commandManager->getNameOfCommand (CommandIDs::openInIDE));
  51. addAndMakeVisible (&saveAndOpenButton);
  52. saveAndOpenButton.setCommandToTrigger (commandManager, CommandIDs::saveAndOpenInIDE, true);
  53. saveAndOpenButton.setButtonText (commandManager->getNameOfCommand (CommandIDs::saveAndOpenInIDE));
  54. #endif
  55. }
  56. void resized()
  57. {
  58. Rectangle<int> r (getAvailableBounds());
  59. r.removeFromBottom (6);
  60. if (saveAndOpenButton.isVisible())
  61. saveAndOpenButton.setBounds (r.removeFromBottom (28).reduced (20, 3));
  62. if (openProjectButton.isVisible())
  63. openProjectButton.setBounds (r.removeFromBottom (28).reduced (20, 3));
  64. tree.setBounds (r);
  65. }
  66. TextButton openProjectButton, saveAndOpenButton;
  67. };
  68. //==============================================================================
  69. ProjectContentComponent::ProjectContentComponent()
  70. : project (nullptr),
  71. currentDocument (nullptr),
  72. treeViewTabs (TabbedButtonBar::TabsAtTop)
  73. {
  74. setOpaque (true);
  75. setWantsKeyboardFocus (true);
  76. treeSizeConstrainer.setMinimumWidth (100);
  77. treeSizeConstrainer.setMaximumWidth (500);
  78. treeViewTabs.setOutline (0);
  79. treeViewTabs.getTabbedButtonBar().setMinimumTabScaleFactor (0.3);
  80. JucerApplication::getApp().openDocumentManager.addListener (this);
  81. }
  82. ProjectContentComponent::~ProjectContentComponent()
  83. {
  84. JucerApplication::getApp().openDocumentManager.removeListener (this);
  85. setProject (nullptr);
  86. contentView = nullptr;
  87. removeChildComponent (&bubbleMessage);
  88. jassert (getNumChildComponents() <= 1);
  89. }
  90. void ProjectContentComponent::paint (Graphics& g)
  91. {
  92. g.fillAll (findColour (mainBackgroundColourId));
  93. }
  94. void ProjectContentComponent::paintOverChildren (Graphics& g)
  95. {
  96. if (contentView != nullptr)
  97. {
  98. const int shadowSize = 15;
  99. const int x = contentView->getX();
  100. ColourGradient cg (Colours::black.withAlpha (0.25f), (float) x, 0,
  101. Colours::transparentBlack, (float) (x - shadowSize), 0, false);
  102. cg.addColour (0.4, Colours::black.withAlpha (0.07f));
  103. cg.addColour (0.6, Colours::black.withAlpha (0.02f));
  104. g.setGradientFill (cg);
  105. g.fillRect (x - shadowSize, 0, shadowSize, getHeight());
  106. }
  107. }
  108. void ProjectContentComponent::resized()
  109. {
  110. Rectangle<int> r (getLocalBounds());
  111. treeViewTabs.setBounds (r.removeFromLeft (treeViewTabs.getWidth()));
  112. if (resizerBar != nullptr)
  113. resizerBar->setBounds (r.removeFromLeft (4));
  114. if (contentView != nullptr)
  115. contentView->setBounds (r);
  116. }
  117. void ProjectContentComponent::lookAndFeelChanged()
  118. {
  119. const Colour tabColour (findColour (mainBackgroundColourId));
  120. for (int i = treeViewTabs.getNumTabs(); --i >= 0;)
  121. treeViewTabs.setTabBackgroundColour (i, tabColour);
  122. repaint();
  123. }
  124. void ProjectContentComponent::childBoundsChanged (Component* child)
  125. {
  126. if (child == &treeViewTabs)
  127. resized();
  128. }
  129. void ProjectContentComponent::setProject (Project* newProject)
  130. {
  131. if (project != newProject)
  132. {
  133. PropertiesFile& settings = getAppProperties();
  134. if (project != nullptr)
  135. project->removeChangeListener (this);
  136. contentView = nullptr;
  137. resizerBar = nullptr;
  138. if (project != nullptr && treeViewTabs.isShowing())
  139. {
  140. if (treeViewTabs.getWidth() > 0)
  141. settings.setValue ("projectTreeviewWidth_" + project->getProjectUID(), treeViewTabs.getWidth());
  142. settings.setValue ("lastTab_" + project->getProjectUID(), treeViewTabs.getCurrentTabName());
  143. }
  144. treeViewTabs.clearTabs();
  145. project = newProject;
  146. if (project != nullptr)
  147. {
  148. addAndMakeVisible (&treeViewTabs);
  149. createProjectTabs();
  150. const String lastTabName (settings.getValue ("lastTab_" + project->getProjectUID()));
  151. int lastTabIndex = treeViewTabs.getTabNames().indexOf (lastTabName);
  152. if (lastTabIndex < 0 || lastTabIndex > treeViewTabs.getNumTabs())
  153. lastTabIndex = 1;
  154. treeViewTabs.setCurrentTabIndex (lastTabIndex);
  155. int lastTreeWidth = settings.getValue ("projectTreeviewWidth_" + project->getProjectUID()).getIntValue();
  156. if (lastTreeWidth < 150)
  157. lastTreeWidth = 250;
  158. treeViewTabs.setBounds (0, 0, lastTreeWidth, getHeight());
  159. addAndMakeVisible (resizerBar = new ResizableEdgeComponent (&treeViewTabs, &treeSizeConstrainer,
  160. ResizableEdgeComponent::rightEdge));
  161. project->addChangeListener (this);
  162. updateMissingFileStatuses();
  163. resized();
  164. }
  165. else
  166. {
  167. treeViewTabs.setVisible (false);
  168. }
  169. }
  170. }
  171. void ProjectContentComponent::createProjectTabs()
  172. {
  173. jassert (project != nullptr);
  174. const Colour tabColour (findColour (mainBackgroundColourId));
  175. treeViewTabs.addTab ("Files", tabColour, new FileTreeTab (*project), true);
  176. treeViewTabs.addTab ("Config", tabColour, new ConfigTreeTab (*project), true);
  177. }
  178. TreeView* ProjectContentComponent::getFilesTreeView() const
  179. {
  180. FileTreeTab* ft = dynamic_cast<FileTreeTab*> (treeViewTabs.getTabContentComponent (0));
  181. return ft != nullptr ? &(ft->tree) : nullptr;
  182. }
  183. ProjectTreeViewBase* ProjectContentComponent::getFilesTreeRoot() const
  184. {
  185. TreeView* tv = getFilesTreeView();
  186. return tv != nullptr ? dynamic_cast <ProjectTreeViewBase*> (tv->getRootItem()) : nullptr;
  187. }
  188. void ProjectContentComponent::saveTreeViewState()
  189. {
  190. for (int i = treeViewTabs.getNumTabs(); --i >= 0;)
  191. {
  192. TreePanelBase* t = dynamic_cast<TreePanelBase*> (treeViewTabs.getTabContentComponent (i));
  193. if (t != nullptr)
  194. t->saveOpenness();
  195. }
  196. }
  197. void ProjectContentComponent::saveOpenDocumentList()
  198. {
  199. if (project != nullptr)
  200. {
  201. ScopedPointer<XmlElement> xml (recentDocumentList.createXML());
  202. if (xml != nullptr)
  203. getAppProperties().setValue ("lastDocs_" + project->getProjectUID(), xml);
  204. }
  205. }
  206. void ProjectContentComponent::reloadLastOpenDocuments()
  207. {
  208. if (project != nullptr)
  209. {
  210. ScopedPointer<XmlElement> xml (getAppProperties().getXmlValue ("lastDocs_" + project->getProjectUID()));
  211. if (xml != nullptr)
  212. {
  213. recentDocumentList.restoreFromXML (*project, *xml);
  214. showDocument (recentDocumentList.getCurrentDocument());
  215. }
  216. }
  217. }
  218. void ProjectContentComponent::documentAboutToClose (OpenDocumentManager::Document* document)
  219. {
  220. hideDocument (document);
  221. }
  222. void ProjectContentComponent::changeListenerCallback (ChangeBroadcaster*)
  223. {
  224. updateMissingFileStatuses();
  225. }
  226. void ProjectContentComponent::updateMissingFileStatuses()
  227. {
  228. ProjectTreeViewBase* p = getFilesTreeRoot();
  229. if (p != nullptr)
  230. p->checkFileStatus();
  231. }
  232. bool ProjectContentComponent::showEditorForFile (const File& f)
  233. {
  234. return getCurrentFile() == f
  235. || showDocument (JucerApplication::getApp().openDocumentManager.openFile (project, f));
  236. }
  237. File ProjectContentComponent::getCurrentFile() const
  238. {
  239. return currentDocument != nullptr ? currentDocument->getFile()
  240. : File::nonexistent;
  241. }
  242. bool ProjectContentComponent::showDocument (OpenDocumentManager::Document* doc)
  243. {
  244. if (doc == nullptr)
  245. return false;
  246. if (doc->hasFileBeenModifiedExternally())
  247. doc->reloadFromFile();
  248. if (doc == getCurrentDocument() && contentView != nullptr)
  249. {
  250. contentView->grabKeyboardFocus();
  251. return true;
  252. }
  253. recentDocumentList.newDocumentOpened (doc);
  254. return setEditorComponent (doc->createEditor(), doc);
  255. }
  256. void ProjectContentComponent::hideEditor()
  257. {
  258. currentDocument = nullptr;
  259. contentView = nullptr;
  260. updateMainWindowTitle();
  261. commandManager->commandStatusChanged();
  262. }
  263. void ProjectContentComponent::hideDocument (OpenDocumentManager::Document* doc)
  264. {
  265. if (doc == currentDocument)
  266. {
  267. OpenDocumentManager::Document* replacement = recentDocumentList.getClosestPreviousDocOtherThan (doc);
  268. if (replacement != nullptr)
  269. showDocument (replacement);
  270. else
  271. hideEditor();
  272. }
  273. }
  274. bool ProjectContentComponent::setEditorComponent (Component* editor, OpenDocumentManager::Document* doc)
  275. {
  276. if (editor != nullptr)
  277. {
  278. contentView = nullptr;
  279. contentView = editor;
  280. currentDocument = doc;
  281. addAndMakeVisible (editor);
  282. resized();
  283. updateMainWindowTitle();
  284. commandManager->commandStatusChanged();
  285. editor->grabKeyboardFocus();
  286. return true;
  287. }
  288. updateMainWindowTitle();
  289. return false;
  290. }
  291. bool ProjectContentComponent::goToPreviousFile()
  292. {
  293. OpenDocumentManager::Document* currentSourceDoc = recentDocumentList.getCurrentDocument();
  294. if (currentSourceDoc != nullptr && currentSourceDoc != getCurrentDocument())
  295. return showDocument (currentSourceDoc);
  296. else
  297. return showDocument (recentDocumentList.getPrevious());
  298. }
  299. bool ProjectContentComponent::goToNextFile()
  300. {
  301. return showDocument (recentDocumentList.getNext());
  302. }
  303. void ProjectContentComponent::updateMainWindowTitle()
  304. {
  305. MainWindow* mw = findParentComponentOfClass<MainWindow>();
  306. if (mw != nullptr)
  307. mw->updateTitle (currentDocument != nullptr ? currentDocument->getName() : String::empty);
  308. }
  309. bool ProjectContentComponent::canProjectBeLaunched() const
  310. {
  311. if (project != nullptr)
  312. {
  313. ScopedPointer <ProjectExporter> launcher (ProjectExporter::createPlatformDefaultExporter (*project));
  314. return launcher != nullptr;
  315. }
  316. return false;
  317. }
  318. ApplicationCommandTarget* ProjectContentComponent::getNextCommandTarget()
  319. {
  320. return findFirstTargetParentComponent();
  321. }
  322. void ProjectContentComponent::getAllCommands (Array <CommandID>& commands)
  323. {
  324. const CommandID ids[] = { CommandIDs::saveDocument,
  325. CommandIDs::closeDocument,
  326. CommandIDs::saveProject,
  327. CommandIDs::closeProject,
  328. CommandIDs::openInIDE,
  329. CommandIDs::saveAndOpenInIDE,
  330. CommandIDs::showFilePanel,
  331. CommandIDs::showConfigPanel,
  332. CommandIDs::goToPreviousDoc,
  333. CommandIDs::goToNextDoc,
  334. StandardApplicationCommandIDs::del };
  335. commands.addArray (ids, numElementsInArray (ids));
  336. }
  337. void ProjectContentComponent::getCommandInfo (const CommandID commandID, ApplicationCommandInfo& result)
  338. {
  339. String documentName;
  340. if (currentDocument != nullptr)
  341. documentName = " '" + currentDocument->getName().substring (0, 32) + "'";
  342. switch (commandID)
  343. {
  344. case CommandIDs::saveProject:
  345. result.setInfo ("Save Project",
  346. "Saves the current project",
  347. CommandCategories::general, 0);
  348. result.setActive (project != nullptr);
  349. break;
  350. case CommandIDs::closeProject:
  351. result.setInfo ("Close Project",
  352. "Closes the current project",
  353. CommandCategories::general, 0);
  354. result.setActive (project != nullptr);
  355. break;
  356. case CommandIDs::saveDocument:
  357. result.setInfo ("Save" + documentName,
  358. "Saves the current document",
  359. CommandCategories::general, 0);
  360. result.setActive (currentDocument != nullptr || project != nullptr);
  361. result.defaultKeypresses.add (KeyPress ('s', ModifierKeys::commandModifier, 0));
  362. break;
  363. case CommandIDs::closeDocument:
  364. result.setInfo ("Close" + documentName,
  365. "Closes the current document",
  366. CommandCategories::general, 0);
  367. result.setActive (currentDocument != nullptr);
  368. #if JUCE_MAC
  369. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier | ModifierKeys::ctrlModifier, 0));
  370. #else
  371. result.defaultKeypresses.add (KeyPress ('w', ModifierKeys::commandModifier | ModifierKeys::shiftModifier, 0));
  372. #endif
  373. break;
  374. case CommandIDs::goToPreviousDoc:
  375. result.setInfo ("Previous Document", "Go to previous document", CommandCategories::general, 0);
  376. result.setActive (recentDocumentList.canGoToPrevious());
  377. #if JUCE_MAC
  378. result.defaultKeypresses.add (KeyPress (KeyPress::leftKey, ModifierKeys::commandModifier | ModifierKeys::ctrlModifier, 0));
  379. #else
  380. result.defaultKeypresses.add (KeyPress (KeyPress::leftKey, ModifierKeys::ctrlModifier | ModifierKeys::shiftModifier, 0));
  381. #endif
  382. break;
  383. case CommandIDs::goToNextDoc:
  384. result.setInfo ("Next Document", "Go to next document", CommandCategories::general, 0);
  385. result.setActive (recentDocumentList.canGoToNext());
  386. #if JUCE_MAC
  387. result.defaultKeypresses.add (KeyPress (KeyPress::rightKey, ModifierKeys::commandModifier | ModifierKeys::ctrlModifier, 0));
  388. #else
  389. result.defaultKeypresses.add (KeyPress (KeyPress::rightKey, ModifierKeys::ctrlModifier | ModifierKeys::shiftModifier, 0));
  390. #endif
  391. break;
  392. case CommandIDs::openInIDE:
  393. #if JUCE_MAC
  394. result.setInfo ("Open in XCode...",
  395. #elif JUCE_WINDOWS
  396. result.setInfo ("Open in Visual Studio...",
  397. #else
  398. result.setInfo ("Open as a Makefile...",
  399. #endif
  400. "Launches the project in an external IDE",
  401. CommandCategories::general, 0);
  402. result.setActive (canProjectBeLaunched());
  403. break;
  404. case CommandIDs::saveAndOpenInIDE:
  405. #if JUCE_MAC
  406. result.setInfo ("Save Project and Open in XCode...",
  407. #elif JUCE_WINDOWS
  408. result.setInfo ("Save Project and Open in Visual Studio...",
  409. #else
  410. result.setInfo ("Save Project and Open as a Makefile...",
  411. #endif
  412. "Saves the project and launches it in an external IDE",
  413. CommandCategories::general, 0);
  414. result.setActive (canProjectBeLaunched());
  415. result.defaultKeypresses.add (KeyPress ('l', ModifierKeys::commandModifier, 0));
  416. break;
  417. case CommandIDs::showFilePanel:
  418. result.setInfo ("Show File Panel",
  419. "Shows the tree of files for this project",
  420. CommandCategories::general, 0);
  421. result.setActive (project != nullptr);
  422. result.defaultKeypresses.add (KeyPress ('p', ModifierKeys::commandModifier, 0));
  423. break;
  424. case CommandIDs::showConfigPanel:
  425. result.setInfo ("Show Config Panel",
  426. "Shows the build options for the project",
  427. CommandCategories::general, 0);
  428. result.setActive (project != nullptr);
  429. result.defaultKeypresses.add (KeyPress ('i', ModifierKeys::commandModifier, 0));
  430. break;
  431. case StandardApplicationCommandIDs::del:
  432. result.setInfo ("Delete Selected File", String::empty, CommandCategories::general, 0);
  433. result.defaultKeypresses.add (KeyPress (KeyPress::deleteKey, 0, 0));
  434. result.defaultKeypresses.add (KeyPress (KeyPress::backspaceKey, 0, 0));
  435. result.setActive (dynamic_cast<TreePanelBase*> (treeViewTabs.getCurrentContentComponent()) != nullptr);
  436. break;
  437. default:
  438. break;
  439. }
  440. }
  441. bool ProjectContentComponent::isCommandActive (const CommandID commandID)
  442. {
  443. return project != nullptr;
  444. }
  445. bool ProjectContentComponent::perform (const InvocationInfo& info)
  446. {
  447. switch (info.commandID)
  448. {
  449. case CommandIDs::saveProject:
  450. if (project != nullptr && ! reinvokeCommandAfterClosingPropertyEditors (info))
  451. project->save (true, true);
  452. break;
  453. case CommandIDs::closeProject:
  454. {
  455. MainWindow* const mw = findParentComponentOfClass<MainWindow>();
  456. if (mw != nullptr && ! reinvokeCommandAfterClosingPropertyEditors (info))
  457. mw->closeCurrentProject();
  458. }
  459. break;
  460. case CommandIDs::saveDocument:
  461. if (! reinvokeCommandAfterClosingPropertyEditors (info))
  462. {
  463. if (currentDocument != nullptr)
  464. currentDocument->save();
  465. else if (project != nullptr)
  466. project->save (true, true);
  467. }
  468. break;
  469. case CommandIDs::closeDocument:
  470. if (currentDocument != nullptr)
  471. JucerApplication::getApp().openDocumentManager.closeDocument (currentDocument, true);
  472. break;
  473. case CommandIDs::goToPreviousDoc:
  474. goToPreviousFile();
  475. break;
  476. case CommandIDs::goToNextDoc:
  477. goToNextFile();
  478. break;
  479. case CommandIDs::openInIDE:
  480. if (project != nullptr)
  481. {
  482. ScopedPointer <ProjectExporter> exporter (ProjectExporter::createPlatformDefaultExporter (*project));
  483. if (exporter != nullptr)
  484. exporter->launchProject();
  485. }
  486. break;
  487. case CommandIDs::saveAndOpenInIDE:
  488. if (project != nullptr)
  489. {
  490. if (! reinvokeCommandAfterClosingPropertyEditors (info))
  491. {
  492. if (project->save (true, true) == FileBasedDocument::savedOk)
  493. {
  494. ScopedPointer <ProjectExporter> exporter (ProjectExporter::createPlatformDefaultExporter (*project));
  495. if (exporter != nullptr)
  496. exporter->launchProject();
  497. }
  498. }
  499. }
  500. break;
  501. case CommandIDs::showFilePanel:
  502. treeViewTabs.setCurrentTabIndex (0);
  503. break;
  504. case CommandIDs::showConfigPanel:
  505. treeViewTabs.setCurrentTabIndex (1);
  506. break;
  507. case StandardApplicationCommandIDs::del:
  508. {
  509. TreePanelBase* const tree = dynamic_cast<TreePanelBase*> (treeViewTabs.getCurrentContentComponent());
  510. if (tree != nullptr)
  511. tree->deleteSelectedItems();
  512. }
  513. break;
  514. default:
  515. return false;
  516. }
  517. return true;
  518. }
  519. bool ProjectContentComponent::reinvokeCommandAfterClosingPropertyEditors (const InvocationInfo& info)
  520. {
  521. if (reinvokeCommandAfterCancellingModalComps (info))
  522. {
  523. grabKeyboardFocus(); // to force any open labels to close their text editors
  524. return true;
  525. }
  526. return false;
  527. }
  528. void ProjectContentComponent::showBubbleMessage (const Rectangle<int>& pos, const String& text)
  529. {
  530. addChildComponent (&bubbleMessage);
  531. bubbleMessage.setAlwaysOnTop (true);
  532. bubbleMessage.showAt (pos, AttributedString (text), 3000, true, false);
  533. }