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.

454 lines
20KB

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