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.

188 lines
7.3KB

  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. #ifndef __JUCER_OPENDOCUMENTMANAGER_JUCEHEADER__
  19. #define __JUCER_OPENDOCUMENTMANAGER_JUCEHEADER__
  20. #include "../Project/jucer_Project.h"
  21. //==============================================================================
  22. /**
  23. */
  24. class OpenDocumentManager
  25. {
  26. public:
  27. //==============================================================================
  28. OpenDocumentManager();
  29. ~OpenDocumentManager();
  30. juce_DeclareSingleton_SingleThreaded_Minimal (OpenDocumentManager);
  31. //==============================================================================
  32. class Document
  33. {
  34. public:
  35. Document() {}
  36. virtual ~Document() {}
  37. virtual bool loadedOk() const = 0;
  38. virtual bool isForFile (const File& file) const = 0;
  39. virtual bool isForNode (const ValueTree& node) const = 0;
  40. virtual bool refersToProject (Project& project) const = 0;
  41. virtual Project* getProject() const = 0;
  42. virtual String getName() const = 0;
  43. virtual String getType() const = 0;
  44. virtual File getFile() const = 0;
  45. virtual bool needsSaving() const = 0;
  46. virtual bool save() = 0;
  47. virtual bool hasFileBeenModifiedExternally() = 0;
  48. virtual void reloadFromFile() = 0;
  49. virtual Component* createEditor() = 0;
  50. virtual Component* createViewer() = 0;
  51. virtual void fileHasBeenRenamed (const File& newFile) = 0;
  52. };
  53. //==============================================================================
  54. int getNumOpenDocuments() const;
  55. Document* getOpenDocument (int index) const;
  56. void moveDocumentToTopOfStack (Document* doc);
  57. bool canOpenFile (const File& file);
  58. Document* openFile (Project* project, const File& file);
  59. bool closeDocument (int index, bool saveIfNeeded);
  60. bool closeDocument (Document* document, bool saveIfNeeded);
  61. bool closeAll (bool askUserToSave);
  62. bool closeAllDocumentsUsingProject (Project& project, bool saveIfNeeded);
  63. void closeFile (const File& f, bool saveIfNeeded);
  64. bool anyFilesNeedSaving() const;
  65. bool saveAll();
  66. FileBasedDocument::SaveResult saveIfNeededAndUserAgrees (Document* doc);
  67. void reloadModifiedFiles();
  68. void fileHasBeenRenamed (const File& oldFile, const File& newFile);
  69. //==============================================================================
  70. class DocumentCloseListener
  71. {
  72. public:
  73. DocumentCloseListener() {}
  74. virtual ~DocumentCloseListener() {}
  75. virtual void documentAboutToClose (Document* document) = 0;
  76. };
  77. void addListener (DocumentCloseListener* listener);
  78. void removeListener (DocumentCloseListener* listener);
  79. //==============================================================================
  80. class DocumentType
  81. {
  82. public:
  83. DocumentType() {}
  84. virtual ~DocumentType() {}
  85. virtual bool canOpenFile (const File& file) = 0;
  86. virtual Document* openFile (Project* project, const File& file) = 0;
  87. };
  88. void registerType (DocumentType* type);
  89. private:
  90. OwnedArray <DocumentType> types;
  91. OwnedArray <Document> documents;
  92. Array <DocumentCloseListener*> listeners;
  93. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (OpenDocumentManager);
  94. };
  95. //==============================================================================
  96. class SourceCodeDocument : public OpenDocumentManager::Document
  97. {
  98. public:
  99. //==============================================================================
  100. SourceCodeDocument (Project* project_, const File& file_)
  101. : modDetector (file_), project (project_)
  102. {
  103. codeDoc = new CodeDocument();
  104. reloadFromFile();
  105. }
  106. //==============================================================================
  107. struct Type : public OpenDocumentManager::DocumentType
  108. {
  109. bool canOpenFile (const File& file) { return file.hasFileExtension ("cpp;h;hpp;mm;m;c;cc;cxx;txt;xml;plist;rtf;html;htm;php;py;rb;cs"); }
  110. Document* openFile (Project* project, const File& file) { return new SourceCodeDocument (project, file); }
  111. };
  112. //==============================================================================
  113. bool loadedOk() const { return true; }
  114. bool isForFile (const File& file) const { return getFile() == file; }
  115. bool isForNode (const ValueTree& node) const { return false; }
  116. bool refersToProject (Project& p) const { return project == &p; }
  117. Project* getProject() const { return project; }
  118. String getName() const { return getFile().getFileName(); }
  119. String getType() const { return getFile().getFileExtension() + " file"; }
  120. File getFile() const { return modDetector.getFile(); }
  121. bool needsSaving() const { return codeDoc != nullptr && codeDoc->hasChangedSinceSavePoint(); }
  122. bool hasFileBeenModifiedExternally() { return modDetector.hasBeenModified(); }
  123. void fileHasBeenRenamed (const File& newFile) { modDetector.fileHasBeenRenamed (newFile); }
  124. void reloadFromFile()
  125. {
  126. modDetector.updateHash();
  127. ScopedPointer <InputStream> in (modDetector.getFile().createInputStream());
  128. if (in != nullptr)
  129. codeDoc->loadFromStream (*in);
  130. }
  131. bool save()
  132. {
  133. TemporaryFile temp (modDetector.getFile());
  134. ScopedPointer <FileOutputStream> out (temp.getFile().createOutputStream());
  135. if (out == nullptr || ! codeDoc->writeToStream (*out))
  136. return false;
  137. out = nullptr;
  138. if (! temp.overwriteTargetFileWithTemporary())
  139. return false;
  140. modDetector.updateHash();
  141. return true;
  142. }
  143. Component* createEditor();
  144. Component* createViewer() { return createEditor(); }
  145. protected:
  146. FileModificationDetector modDetector;
  147. ScopedPointer <CodeDocument> codeDoc;
  148. Project* project;
  149. };
  150. #endif // __JUCER_OPENDOCUMENTMANAGER_JUCEHEADER__