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.

488 lines
20KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. #pragma once
  20. #include "../Project/jucer_Project.h"
  21. #include "../Utility/UI/PropertyComponents/jucer_PropertyComponentsWithEnablement.h"
  22. class ProjectSaver;
  23. //==============================================================================
  24. class ProjectExporter : private Value::Listener
  25. {
  26. public:
  27. ProjectExporter (Project&, const ValueTree& settings);
  28. virtual ~ProjectExporter() override;
  29. struct ExporterTypeInfo
  30. {
  31. String name;
  32. const void* iconData;
  33. int iconDataSize;
  34. Image getIcon() const
  35. {
  36. Image image (Image::ARGB, 200, 200, true);
  37. Graphics g (image);
  38. std::unique_ptr<Drawable> svgDrawable (Drawable::createFromImageData (iconData, (size_t) iconDataSize));
  39. svgDrawable->drawWithin (g, image.getBounds().toFloat(), RectanglePlacement::fillDestination, 1.0f);
  40. return image;
  41. }
  42. };
  43. static StringArray getExporterNames();
  44. static StringArray getExporterValueTreeNames();
  45. static Array<ExporterTypeInfo> getExporterTypes();
  46. static String getValueTreeNameForExporter (const String& exporterName);
  47. static String getTargetFolderForExporter (const String& exporterValueTreeName);
  48. static StringArray getAllDefaultBuildsFolders();
  49. static ProjectExporter* createNewExporter (Project&, const int index);
  50. static ProjectExporter* createNewExporter (Project&, const String& name);
  51. static ProjectExporter* createExporter (Project&, const ValueTree& settings);
  52. static bool canProjectBeLaunched (Project*);
  53. static String getCurrentPlatformExporterName();
  54. //==============================================================================
  55. // capabilities of exporter
  56. virtual bool usesMMFiles() const = 0;
  57. virtual void createExporterProperties (PropertyListBuilder&) = 0;
  58. virtual bool canLaunchProject() = 0;
  59. virtual bool launchProject() = 0;
  60. virtual void create (const OwnedArray<LibraryModule>&) const = 0; // may throw a SaveError
  61. virtual bool shouldFileBeCompiledByDefault (const File& path) const;
  62. virtual bool canCopeWithDuplicateFiles() = 0;
  63. virtual bool supportsUserDefinedConfigurations() const = 0; // false if exporter only supports two configs Debug and Release
  64. virtual void updateDeprecatedProjectSettingsInteractively();
  65. virtual void initialiseDependencyPathValues() {}
  66. // IDE targeted by exporter
  67. virtual bool isXcode() const = 0;
  68. virtual bool isVisualStudio() const = 0;
  69. virtual bool isCodeBlocks() const = 0;
  70. virtual bool isMakefile() const = 0;
  71. virtual bool isAndroidStudio() const = 0;
  72. virtual bool isCLion() const = 0;
  73. // operating system targeted by exporter
  74. virtual bool isAndroid() const = 0;
  75. virtual bool isWindows() const = 0;
  76. virtual bool isLinux() const = 0;
  77. virtual bool isOSX() const = 0;
  78. virtual bool isiOS() const = 0;
  79. virtual String getDescription() { return {}; }
  80. //==============================================================================
  81. // cross-platform audio plug-ins supported by exporter
  82. virtual bool supportsTargetType (ProjectType::Target::Type type) const = 0;
  83. inline bool shouldBuildTargetType (ProjectType::Target::Type type) const
  84. {
  85. return project.shouldBuildTargetType (type) && supportsTargetType (type);
  86. }
  87. inline void callForAllSupportedTargets (std::function<void (ProjectType::Target::Type)> callback)
  88. {
  89. for (int i = 0; i < ProjectType::Target::unspecified; ++i)
  90. if (shouldBuildTargetType (static_cast<ProjectType::Target::Type> (i)))
  91. callback (static_cast<ProjectType::Target::Type> (i));
  92. }
  93. //==============================================================================
  94. bool mayCompileOnCurrentOS() const
  95. {
  96. #if JUCE_MAC
  97. return isOSX() || isAndroid() || isiOS();
  98. #elif JUCE_WINDOWS
  99. return isWindows() || isAndroid();
  100. #elif JUCE_LINUX
  101. return isLinux() || isAndroid();
  102. #else
  103. #error
  104. #endif
  105. }
  106. //==============================================================================
  107. String getName() const;
  108. File getTargetFolder() const;
  109. Project& getProject() noexcept { return project; }
  110. const Project& getProject() const noexcept { return project; }
  111. UndoManager* getUndoManager() const { return project.getUndoManagerFor (settings); }
  112. Value getSetting (const Identifier& nm) { return settings.getPropertyAsValue (nm, project.getUndoManagerFor (settings)); }
  113. String getSettingString (const Identifier& nm) const { return settings [nm]; }
  114. Value getTargetLocationValue() { return targetLocationValue.getPropertyAsValue(); }
  115. String getTargetLocationString() const { return targetLocationValue.get(); }
  116. String getExtraCompilerFlagsString() const { return extraCompilerFlagsValue.get().toString().replaceCharacters ("\r\n", " "); }
  117. String getExtraLinkerFlagsString() const { return extraLinkerFlagsValue.get().toString().replaceCharacters ("\r\n", " "); }
  118. String getExternalLibrariesString() const { return getSearchPathsFromString (externalLibrariesValue.get().toString()).joinIntoString (";"); }
  119. bool shouldUseGNUExtensions() const { return gnuExtensionsValue.get(); }
  120. String getVSTLegacyPathString() const { return vstLegacyPathValueWrapper.wrappedValue.get(); }
  121. String getVST3PathString() const { return vst3PathValueWrapper.wrappedValue.get(); }
  122. String getAAXPathString() const { return aaxPathValueWrapper.wrappedValue.get(); }
  123. String getRTASPathString() const { return rtasPathValueWrapper.wrappedValue.get(); }
  124. // NB: this is the path to the parent "modules" folder that contains the named module, not the
  125. // module folder itself.
  126. ValueWithDefault getPathForModuleValue (const String& moduleID);
  127. String getPathForModuleString (const String& moduleID) const;
  128. void removePathForModule (const String& moduleID);
  129. TargetOS::OS getTargetOSForExporter() const;
  130. RelativePath getLegacyModulePath (const String& moduleID) const;
  131. String getLegacyModulePath() const;
  132. // Returns a path to the actual module folder itself
  133. RelativePath getModuleFolderRelativeToProject (const String& moduleID) const;
  134. void updateOldModulePaths();
  135. RelativePath rebaseFromProjectFolderToBuildTarget (const RelativePath& path) const;
  136. void addToExtraSearchPaths (const RelativePath& pathFromProjectFolder, int index = -1);
  137. void addToModuleLibPaths (const RelativePath& pathFromProjectFolder);
  138. void addProjectPathToBuildPathList (StringArray&, const RelativePath&, int index = -1) const;
  139. Drawable* getBigIcon() const;
  140. Drawable* getSmallIcon() const;
  141. Image getBestIconForSize (int size, bool returnNullIfNothingBigEnough) const;
  142. String getExporterIdentifierMacro() const
  143. {
  144. return "JUCER_" + settings.getType().toString() + "_"
  145. + String::toHexString (getTargetLocationString().hashCode()).toUpperCase();
  146. }
  147. // An exception that can be thrown by the create() method.
  148. class SaveError
  149. {
  150. public:
  151. SaveError (const String& error) : message (error)
  152. {}
  153. SaveError (const File& fileThatFailedToWrite)
  154. : message ("Can't write to the file: " + fileThatFailedToWrite.getFullPathName())
  155. {}
  156. String message;
  157. };
  158. void createPropertyEditors (PropertyListBuilder&);
  159. void addSettingsForProjectType (const ProjectType&);
  160. //==============================================================================
  161. void copyMainGroupFromProject();
  162. Array<Project::Item>& getAllGroups() noexcept { jassert (itemGroups.size() > 0); return itemGroups; }
  163. const Array<Project::Item>& getAllGroups() const noexcept { jassert (itemGroups.size() > 0); return itemGroups; }
  164. Project::Item& getModulesGroup();
  165. //==============================================================================
  166. StringArray linuxLibs, linuxPackages, makefileExtraLinkerFlags;
  167. //==============================================================================
  168. StringPairArray msvcExtraPreprocessorDefs;
  169. String msvcDelayLoadedDLLs;
  170. StringArray mingwLibs, windowsLibs;
  171. //==============================================================================
  172. StringArray androidLibs;
  173. //==============================================================================
  174. StringArray extraSearchPaths;
  175. StringArray moduleLibSearchPaths;
  176. //==============================================================================
  177. class BuildConfiguration : public ReferenceCountedObject
  178. {
  179. public:
  180. BuildConfiguration (Project& project, const ValueTree& configNode, const ProjectExporter&);
  181. ~BuildConfiguration();
  182. using Ptr = ReferenceCountedObjectPtr<BuildConfiguration>;
  183. //==============================================================================
  184. virtual void createConfigProperties (PropertyListBuilder&) = 0;
  185. virtual String getModuleLibraryArchName() const = 0;
  186. //==============================================================================
  187. String getName() const { return configNameValue.get(); }
  188. bool isDebug() const { return isDebugValue.get(); }
  189. String getTargetBinaryRelativePathString() const { return targetBinaryPathValue.get(); }
  190. String getTargetBinaryNameString (bool isUnityPlugin = false) const
  191. {
  192. return (isUnityPlugin ? Project::addUnityPluginPrefixIfNecessary (targetNameValue.get().toString())
  193. : targetNameValue.get().toString());
  194. }
  195. int getOptimisationLevelInt() const { return optimisationLevelValue.get(); }
  196. String getGCCOptimisationFlag() const;
  197. bool isLinkTimeOptimisationEnabled() const { return linkTimeOptimisationValue.get(); }
  198. String getBuildConfigPreprocessorDefsString() const { return ppDefinesValue.get(); }
  199. StringPairArray getAllPreprocessorDefs() const; // includes inherited definitions
  200. StringPairArray getUniquePreprocessorDefs() const; // returns pre-processor definitions that are not already in the project pre-processor defs
  201. String getHeaderSearchPathString() const { return headerSearchPathValue.get(); }
  202. StringArray getHeaderSearchPaths() const;
  203. String getLibrarySearchPathString() const { return librarySearchPathValue.get(); }
  204. StringArray getLibrarySearchPaths() const;
  205. String getGCCLibraryPathFlags() const;
  206. //==============================================================================
  207. Value getValue (const Identifier& nm) { return config.getPropertyAsValue (nm, getUndoManager()); }
  208. UndoManager* getUndoManager() const { return project.getUndoManagerFor (config); }
  209. //==============================================================================
  210. void createPropertyEditors (PropertyListBuilder&);
  211. void addGCCOptimisationProperty (PropertyListBuilder&);
  212. void removeFromExporter();
  213. //==============================================================================
  214. ValueTree config;
  215. Project& project;
  216. const ProjectExporter& exporter;
  217. protected:
  218. ValueWithDefault isDebugValue, configNameValue, targetNameValue, targetBinaryPathValue, optimisationLevelValue,
  219. linkTimeOptimisationValue, ppDefinesValue, headerSearchPathValue, librarySearchPathValue, userNotesValue;
  220. private:
  221. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (BuildConfiguration)
  222. };
  223. void addNewConfigurationFromExisting (const BuildConfiguration& configToCopy);
  224. void addNewConfiguration (bool isDebugConfig);
  225. bool hasConfigurationNamed (const String& name) const;
  226. String getUniqueConfigName (String name) const;
  227. String getExternalLibraryFlags (const BuildConfiguration& config) const;
  228. //==============================================================================
  229. struct ConfigIterator
  230. {
  231. ConfigIterator (ProjectExporter& exporter);
  232. bool next();
  233. BuildConfiguration& operator*() const { return *config; }
  234. BuildConfiguration* operator->() const { return config.get(); }
  235. BuildConfiguration::Ptr config;
  236. int index;
  237. private:
  238. ProjectExporter& exporter;
  239. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ConfigIterator)
  240. };
  241. struct ConstConfigIterator
  242. {
  243. ConstConfigIterator (const ProjectExporter& exporter);
  244. bool next();
  245. const BuildConfiguration& operator*() const { return *config; }
  246. const BuildConfiguration* operator->() const { return config.get(); }
  247. BuildConfiguration::Ptr config;
  248. int index;
  249. private:
  250. const ProjectExporter& exporter;
  251. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ConstConfigIterator)
  252. };
  253. int getNumConfigurations() const;
  254. BuildConfiguration::Ptr getConfiguration (int index) const;
  255. ValueTree getConfigurations() const;
  256. virtual void createDefaultConfigs();
  257. void createDefaultModulePaths();
  258. //==============================================================================
  259. Value getExporterPreprocessorDefsValue() { return extraPPDefsValue.getPropertyAsValue(); }
  260. String getExporterPreprocessorDefsString() const { return extraPPDefsValue.get(); }
  261. // includes exporter, project + config defs
  262. StringPairArray getAllPreprocessorDefs (const BuildConfiguration& config, const ProjectType::Target::Type targetType) const;
  263. // includes exporter + project defs..
  264. StringPairArray getAllPreprocessorDefs() const;
  265. void addTargetSpecificPreprocessorDefs (StringPairArray& defs, const ProjectType::Target::Type targetType) const;
  266. String replacePreprocessorTokens (const BuildConfiguration&, const String& sourceString) const;
  267. ValueTree settings;
  268. enum GCCOptimisationLevel
  269. {
  270. gccO0 = 1,
  271. gccO1 = 4,
  272. gccO2 = 5,
  273. gccO3 = 3,
  274. gccOs = 2,
  275. gccOfast = 6
  276. };
  277. protected:
  278. //==============================================================================
  279. String name;
  280. Project& project;
  281. const ProjectType& projectType;
  282. const String projectName;
  283. const File projectFolder;
  284. //==============================================================================
  285. // Wraps a ValueWithDefault object that has a default which depends on a global value.
  286. // Used for the VST3, RTAS and AAX project-specific path options.
  287. struct ValueWithDefaultWrapper : public Value::Listener
  288. {
  289. void init (const ValueWithDefault& vwd, ValueWithDefault global, TargetOS::OS targetOS)
  290. {
  291. wrappedValue = vwd;
  292. globalValue = global.getPropertyAsValue();
  293. globalIdentifier = global.getPropertyID();
  294. os = targetOS;
  295. if (wrappedValue.get() == var())
  296. wrappedValue.resetToDefault();
  297. globalValue.addListener (this);
  298. valueChanged (globalValue);
  299. }
  300. void valueChanged (Value&) override
  301. {
  302. wrappedValue.setDefault (getAppSettings().getStoredPath (globalIdentifier, os).get());
  303. }
  304. ValueWithDefault wrappedValue;
  305. Value globalValue;
  306. Identifier globalIdentifier;
  307. TargetOS::OS os;
  308. };
  309. ValueWithDefaultWrapper vstLegacyPathValueWrapper, vst3PathValueWrapper, rtasPathValueWrapper, aaxPathValueWrapper;
  310. ValueWithDefault targetLocationValue, extraCompilerFlagsValue, extraLinkerFlagsValue, externalLibrariesValue,
  311. userNotesValue, gnuExtensionsValue, bigIconValue, smallIconValue, extraPPDefsValue;
  312. Value projectCompilerFlagSchemesValue;
  313. HashMap<String, ValueWithDefault> compilerFlagSchemesMap;
  314. mutable Array<Project::Item> itemGroups;
  315. void initItemGroups() const;
  316. Project::Item* modulesGroup = nullptr;
  317. virtual BuildConfiguration::Ptr createBuildConfig (const ValueTree&) const = 0;
  318. void addDefaultPreprocessorDefs (StringPairArray&) const;
  319. static String getDefaultBuildsRootFolder() { return "Builds/"; }
  320. static String getStaticLibbedFilename (String name) { return addSuffix (addLibPrefix (name), ".a"); }
  321. static String getDynamicLibbedFilename (String name) { return addSuffix (addLibPrefix (name), ".so"); }
  322. virtual void addPlatformSpecificSettingsForProjectType (const ProjectType&) = 0;
  323. //==============================================================================
  324. static void overwriteFileIfDifferentOrThrow (const File& file, const MemoryOutputStream& newData)
  325. {
  326. if (! FileHelpers::overwriteFileWithNewDataIfDifferent (file, newData))
  327. throw SaveError (file);
  328. }
  329. static void overwriteFileIfDifferentOrThrow (const File& file, const String& newData)
  330. {
  331. if (! FileHelpers::overwriteFileWithNewDataIfDifferent (file, newData))
  332. throw SaveError (file);
  333. }
  334. static void createDirectoryOrThrow (const File& dirToCreate)
  335. {
  336. if (! dirToCreate.createDirectory())
  337. throw SaveError ("Can't create folder: " + dirToCreate.getFullPathName());
  338. }
  339. static void writeXmlOrThrow (const XmlElement& xml, const File& file, const String& encoding, int maxCharsPerLine, bool useUnixNewLines = false)
  340. {
  341. MemoryOutputStream mo;
  342. if (useUnixNewLines)
  343. mo.setNewLineString ("\n");
  344. xml.writeToStream (mo, String(), false, true, encoding, maxCharsPerLine);
  345. overwriteFileIfDifferentOrThrow (file, mo);
  346. }
  347. static Image rescaleImageForIcon (Drawable&, int iconSize);
  348. private:
  349. //==============================================================================
  350. void valueChanged (Value&) override { updateCompilerFlagValues(); }
  351. void updateCompilerFlagValues();
  352. //==============================================================================
  353. static String addLibPrefix (const String name)
  354. {
  355. return name.startsWith ("lib") ? name
  356. : "lib" + name;
  357. }
  358. static String addSuffix (const String name, const String suffix)
  359. {
  360. return name.endsWithIgnoreCase (suffix) ? name
  361. : name + suffix;
  362. }
  363. void createDependencyPathProperties (PropertyListBuilder&);
  364. void createIconProperties (PropertyListBuilder&);
  365. void addVSTPathsIfPluginOrHost();
  366. void addCommonAudioPluginSettings();
  367. void addLegacyVSTFolderToPathIfSpecified();
  368. RelativePath getInternalVST3SDKPath();
  369. void addVST3FolderToPath();
  370. void addAAXFoldersToPath();
  371. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ProjectExporter)
  372. };