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.

684 lines
23KB

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