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.

298 lines
9.3KB

  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. #include "../jucer_Headers.h"
  20. #include "jucer_ResourceFile.h"
  21. #include "../Application/jucer_OpenDocumentManager.h"
  22. static const char* resourceFileIdentifierString = "JUCER_BINARY_RESOURCE";
  23. //==============================================================================
  24. ResourceFile::ResourceFile (Project& p)
  25. : project (p),
  26. className ("BinaryData")
  27. {
  28. addResourcesFromProjectItem (project.getMainGroup());
  29. }
  30. ResourceFile::~ResourceFile()
  31. {
  32. }
  33. //==============================================================================
  34. void ResourceFile::addResourcesFromProjectItem (const Project::Item& projectItem)
  35. {
  36. if (projectItem.isGroup())
  37. {
  38. for (int i = 0; i < projectItem.getNumChildren(); ++i)
  39. addResourcesFromProjectItem (projectItem.getChild(i));
  40. }
  41. else
  42. {
  43. if (projectItem.shouldBeAddedToBinaryResources())
  44. addFile (projectItem.getFile());
  45. }
  46. }
  47. //==============================================================================
  48. void ResourceFile::setClassName (const String& name)
  49. {
  50. className = name;
  51. }
  52. void ResourceFile::addFile (const File& file)
  53. {
  54. files.add (file);
  55. const String variableNameRoot (CodeHelpers::makeBinaryDataIdentifierName (file));
  56. String variableName (variableNameRoot);
  57. int suffix = 2;
  58. while (variableNames.contains (variableName))
  59. variableName = variableNameRoot + String (suffix++);
  60. variableNames.add (variableName);
  61. }
  62. String ResourceFile::getDataVariableFor (const File& file) const
  63. {
  64. jassert (files.indexOf (file) >= 0);
  65. return variableNames [files.indexOf (file)];
  66. }
  67. String ResourceFile::getSizeVariableFor (const File& file) const
  68. {
  69. jassert (files.indexOf (file) >= 0);
  70. return variableNames [files.indexOf (file)] + "Size";
  71. }
  72. int64 ResourceFile::getTotalDataSize() const
  73. {
  74. int64 total = 0;
  75. for (int i = 0; i < files.size(); ++i)
  76. total += files.getReference(i).getSize();
  77. return total;
  78. }
  79. static String getComment()
  80. {
  81. String comment;
  82. comment << newLine << newLine
  83. << " This is an auto-generated file: Any edits you make may be overwritten!" << newLine
  84. << newLine
  85. << "*/" << newLine
  86. << newLine;
  87. return comment;
  88. }
  89. Result ResourceFile::writeHeader (MemoryOutputStream& header)
  90. {
  91. header << "/* ========================================================================================="
  92. << getComment()
  93. << "#pragma once" << newLine
  94. << newLine
  95. << "namespace " << className << newLine
  96. << "{" << newLine;
  97. bool containsAnyImages = false;
  98. for (int i = 0; i < files.size(); ++i)
  99. {
  100. const File& file = files.getReference(i);
  101. if (! file.existsAsFile())
  102. return Result::fail ("Can't open resource file: " + file.getFullPathName());
  103. const int64 dataSize = file.getSize();
  104. const String variableName (variableNames[i]);
  105. FileInputStream fileStream (file);
  106. if (fileStream.openedOk())
  107. {
  108. containsAnyImages = containsAnyImages
  109. || (ImageFileFormat::findImageFormatForStream (fileStream) != nullptr);
  110. header << " extern const char* " << variableName << ";" << newLine;
  111. header << " const int " << variableName << "Size = " << (int) dataSize << ";" << newLine << newLine;
  112. }
  113. }
  114. header << " // Points to the start of a list of resource names." << newLine
  115. << " extern const char* namedResourceList[];" << newLine
  116. << newLine
  117. << " // Number of elements in the namedResourceList array." << newLine
  118. << " const int namedResourceListSize = " << files.size() << ";" << newLine
  119. << newLine
  120. << " // If you provide the name of one of the binary resource variables above, this function will" << newLine
  121. << " // return the corresponding data and its size (or a null pointer if the name isn't found)." << newLine
  122. << " const char* getNamedResource (const char* resourceNameUTF8, int& dataSizeInBytes) throw();" << newLine
  123. << "}" << newLine;
  124. return Result::ok();
  125. }
  126. Result ResourceFile::writeCpp (MemoryOutputStream& cpp, const File& headerFile, int& i, const int maxFileSize)
  127. {
  128. const bool isFirstFile = (i == 0);
  129. cpp << "/* ==================================== " << resourceFileIdentifierString << " ===================================="
  130. << getComment()
  131. << "namespace " << className << newLine
  132. << "{" << newLine;
  133. bool containsAnyImages = false;
  134. while (i < files.size())
  135. {
  136. const File& file = files.getReference(i);
  137. const String variableName (variableNames[i]);
  138. FileInputStream fileStream (file);
  139. if (fileStream.openedOk())
  140. {
  141. containsAnyImages = containsAnyImages
  142. || (ImageFileFormat::findImageFormatForStream (fileStream) != nullptr);
  143. const String tempVariable ("temp_binary_data_" + String (i));
  144. cpp << newLine << "//================== " << file.getFileName() << " ==================" << newLine
  145. << "static const unsigned char " << tempVariable << "[] =" << newLine;
  146. {
  147. MemoryBlock data;
  148. fileStream.readIntoMemoryBlock (data);
  149. CodeHelpers::writeDataAsCppLiteral (data, cpp, true, true);
  150. }
  151. cpp << newLine << newLine
  152. << "const char* " << variableName << " = (const char*) " << tempVariable << ";" << newLine;
  153. }
  154. ++i;
  155. if (cpp.getPosition() > maxFileSize)
  156. break;
  157. }
  158. if (isFirstFile)
  159. {
  160. if (i < files.size())
  161. {
  162. cpp << newLine
  163. << "}" << newLine
  164. << newLine
  165. << "#include \"" << headerFile.getFileName() << "\"" << newLine
  166. << newLine
  167. << "namespace " << className << newLine
  168. << "{";
  169. }
  170. cpp << newLine
  171. << newLine
  172. << "const char* getNamedResource (const char*, int&) throw();" << newLine
  173. << "const char* getNamedResource (const char* resourceNameUTF8, int& numBytes) throw()" << newLine
  174. << "{" << newLine;
  175. StringArray returnCodes;
  176. for (int j = 0; j < files.size(); ++j)
  177. {
  178. const File& file = files.getReference(j);
  179. const int64 dataSize = file.getSize();
  180. returnCodes.add ("numBytes = " + String (dataSize) + "; return " + variableNames[j] + ";");
  181. }
  182. CodeHelpers::createStringMatcher (cpp, "resourceNameUTF8", variableNames, returnCodes, 4);
  183. cpp << " numBytes = 0;" << newLine
  184. << " return 0;" << newLine
  185. << "}" << newLine
  186. << newLine
  187. << "const char* namedResourceList[] =" << newLine
  188. << "{" << newLine;
  189. for (int j = 0; j < files.size(); ++j)
  190. cpp << " " << variableNames[j].quoted() << (j < files.size() - 1 ? "," : "") << newLine;
  191. cpp << "};" << newLine;
  192. }
  193. cpp << newLine
  194. << "}" << newLine;
  195. return Result::ok();
  196. }
  197. Result ResourceFile::write (Array<File>& filesCreated, const int maxFileSize)
  198. {
  199. const File headerFile (project.getBinaryDataHeaderFile());
  200. {
  201. MemoryOutputStream mo;
  202. Result r (writeHeader (mo));
  203. if (r.failed())
  204. return r;
  205. if (! FileHelpers::overwriteFileWithNewDataIfDifferent (headerFile, mo))
  206. return Result::fail ("Can't write to file: " + headerFile.getFullPathName());
  207. filesCreated.add (headerFile);
  208. }
  209. int i = 0;
  210. int fileIndex = 0;
  211. for (;;)
  212. {
  213. File cpp (project.getBinaryDataCppFile (fileIndex));
  214. MemoryOutputStream mo;
  215. Result r (writeCpp (mo, headerFile, i, maxFileSize));
  216. if (r.failed())
  217. return r;
  218. if (! FileHelpers::overwriteFileWithNewDataIfDifferent (cpp, mo))
  219. return Result::fail ("Can't write to file: " + cpp.getFullPathName());
  220. filesCreated.add (cpp);
  221. ++fileIndex;
  222. if (i >= files.size())
  223. break;
  224. }
  225. return Result::ok();
  226. }