/* ============================================================================== This file is part of the JUCE library - "Jules' Utility Class Extensions" Copyright 2004-10 by Raw Material Software Ltd. ------------------------------------------------------------------------------ JUCE can be redistributed and/or modified under the terms of the GNU General Public License (Version 2), as published by the Free Software Foundation. A copy of the license is included in the JUCE distribution, or can be found online at www.gnu.org/licenses. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. ------------------------------------------------------------------------------ To release a closed-source product which uses JUCE, commercial licenses are available: visit www.rawmaterialsoftware.com/juce for more information. ============================================================================== */ #include "jucer_ResourceFile.h" #include "jucer_ProjectTreeViewBase.h" #include "../Application/jucer_OpenDocumentManager.h" static const char* resourceFileIdentifierString = "JUCER_BINARY_RESOURCE"; //============================================================================== ResourceFile::ResourceFile (Project& project_) : project (project_), className ("BinaryData") { addResourcesFromProjectItem (project.getMainGroup()); } ResourceFile::~ResourceFile() { } bool ResourceFile::isResourceFile (const File& file) { if (file.hasFileExtension ("cpp;cc;h")) { ScopedPointer in (file.createInputStream()); if (in != 0) { MemoryBlock mb; in->readIntoMemoryBlock (mb, 256); return mb.toString().contains (resourceFileIdentifierString); } } return false; } //============================================================================== void ResourceFile::addResourcesFromProjectItem (const Project::Item& projectItem) { if (projectItem.isGroup()) { for (int i = 0; i < projectItem.getNumChildren(); ++i) addResourcesFromProjectItem (projectItem.getChild(i)); } else { if (projectItem.shouldBeAddedToBinaryResources()) addFile (projectItem.getFile()); } } //============================================================================== void ResourceFile::setJuceHeaderToInclude (const File& header) { juceHeader = header; } void ResourceFile::setClassName (const String& className_) { className = className_; } void ResourceFile::addFile (const File& file) { files.add (new File (file)); } int64 ResourceFile::getTotalDataSize() const { int64 total = 0; for (int i = 0; i < files.size(); ++i) total += files.getUnchecked(i)->getSize(); return total; } bool ResourceFile::write (const File& cppFile, OutputStream& cpp, OutputStream& header) { String comment; comment << newLine << newLine << " This is an auto-generated file, created by " << JUCEApplication::getInstance()->getApplicationName() << newLine << " Do not edit anything in this file!" << newLine << newLine << "*/" << newLine << newLine; header << "/* =========================================================================================" << comment; cpp << "/* ==================================== " << resourceFileIdentifierString << " ====================================" << comment; if (juceHeader.exists()) header << CodeHelpers::createIncludeStatement (juceHeader, cppFile) << newLine; const String namespaceName (className); StringArray variableNames, returnCodes; int i; for (i = 0; i < files.size(); ++i) { String variableNameRoot (CodeHelpers::makeValidIdentifier (files.getUnchecked(i)->getFileName() .replaceCharacters (" .", "__") .retainCharacters ("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz_0123456789"), false, true, false)); String variableName (variableNameRoot); int suffix = 2; while (variableNames.contains (variableName)) variableName = variableNameRoot + String (suffix++); variableNames.add (variableName); returnCodes.add ("numBytes = " + namespaceName + "::" + variableName + "Size; return " + namespaceName + "::" + variableName + ";"); } cpp << CodeHelpers::createIncludeStatement (cppFile.withFileExtension (".h"), cppFile) << newLine << newLine << newLine << "const char* " << namespaceName << "::getNamedResource (const char* resourceNameUTF8, int& numBytes) throw()" << newLine << "{" << newLine; CodeHelpers::createStringMatcher (cpp, "resourceNameUTF8", variableNames, returnCodes, 4); cpp << " numBytes = 0;" << newLine << " return 0;" << newLine << "}" << newLine << newLine; header << "namespace " << namespaceName << newLine << "{" << newLine; for (i = 0; i < files.size(); ++i) { const File file (*files.getUnchecked(i)); const int64 dataSize = file.getSize(); ScopedPointer fileStream (file.createInputStream()); jassert (fileStream != 0); if (fileStream != 0) { const String variableName (variableNames[i]); const String tempVariable ("temp_" + String::toHexString (file.hashCode())); header << " extern const char* " << variableName << ";" << newLine; header << " const int " << variableName << "Size = " << (int) dataSize << ";" << newLine << newLine; cpp << newLine << "//================== " << file.getFileName() << " ==================" << newLine << "static const unsigned char " << tempVariable << "[] =" << newLine; { MemoryBlock data; fileStream->readIntoMemoryBlock (data); CodeHelpers::writeDataAsCppLiteral (data, cpp); } cpp << newLine << newLine << "const char* " << namespaceName << "::" << variableName << " = (const char*) " << tempVariable << ";" << newLine; } } header << " // If you provide the name of one of the binary resource variables above, this function will" << newLine << " // return the corresponding data and its size (or a null pointer if the name isn't found)." << newLine << " const char* getNamedResource (const char* resourceNameUTF8, int& dataSizeInBytes) throw();" << newLine << "}" << newLine; return true; } bool ResourceFile::write (const File& cppFile) { TemporaryFile tempH (cppFile.withFileExtension (".h"), TemporaryFile::useHiddenFile); TemporaryFile tempCpp (cppFile, TemporaryFile::useHiddenFile); ScopedPointer cppOut (tempCpp.getFile().createOutputStream (32768)); ScopedPointer hppOut (tempH.getFile().createOutputStream (32768)); if (cppOut != 0 && hppOut != 0) { if (write (cppFile, *cppOut, *hppOut)) { cppOut = 0; hppOut = 0; return (tempCpp.getFile().hasIdenticalContentTo (tempCpp.getTargetFile()) || tempCpp.overwriteTargetFileWithTemporary()) && (tempH.getFile().hasIdenticalContentTo (tempH.getTargetFile()) || tempH.overwriteTargetFileWithTemporary()); } } return false; }