/* ============================================================================== This file is part of the JUCE library - "Jules' Utility Class Extensions" Copyright 2004-11 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_Headers.h" #include "jucer_CodeHelpers.h" //============================================================================== namespace FileHelpers { int64 calculateMemoryHashCode (const void* data, const int numBytes) { int64 t = 0; for (int i = 0; i < numBytes; ++i) t = t * 65599 + static_cast (data)[i]; return t; } int64 calculateStreamHashCode (InputStream& in) { int64 t = 0; const int bufferSize = 4096; HeapBlock buffer; buffer.malloc (bufferSize); for (;;) { const int num = in.read (buffer, bufferSize); if (num <= 0) break; for (int i = 0; i < num; ++i) t = t * 65599 + buffer[i]; } return t; } int64 calculateFileHashCode (const File& file) { ScopedPointer stream (file.createInputStream()); return stream != nullptr ? calculateStreamHashCode (*stream) : 0; } bool overwriteFileWithNewDataIfDifferent (const File& file, const void* data, int numBytes) { if (file.getSize() == numBytes && calculateMemoryHashCode (data, numBytes) == calculateFileHashCode (file)) return true; if (file.exists()) return file.replaceWithData (data, numBytes); else return file.appendData (data, numBytes); } bool overwriteFileWithNewDataIfDifferent (const File& file, const MemoryOutputStream& newData) { return overwriteFileWithNewDataIfDifferent (file, newData.getData(), newData.getDataSize()); } bool overwriteFileWithNewDataIfDifferent (const File& file, const String& newData) { const char* const utf8 = newData.toUTF8(); return overwriteFileWithNewDataIfDifferent (file, utf8, strlen (utf8)); } bool containsAnyNonHiddenFiles (const File& folder) { DirectoryIterator di (folder, false); while (di.next()) if (! di.getFile().isHidden()) return true; return false; } String unixStylePath (const String& path) { return path.replaceCharacter ('\\', '/'); } String windowsStylePath (const String& path) { return path.replaceCharacter ('/', '\\'); } String appendPath (const String& path, const String& subpath) { if (File::isAbsolutePath (subpath) || subpath.startsWithChar ('$') || subpath.startsWithChar ('~') || (CharacterFunctions::isLetter (subpath[0]) && subpath[1] == ':')) return subpath.replaceCharacter ('\\', '/'); String path1 (path.replaceCharacter ('\\', '/')); if (! path1.endsWithChar ('/')) path1 << '/'; return path1 + subpath.replaceCharacter ('\\', '/'); } bool shouldPathsBeRelative (String path1, String path2) { path1 = unixStylePath (path1); path2 = unixStylePath (path2); const int len = jmin (path1.length(), path2.length()); int commonBitLength = 0; for (int i = 0; i < len; ++i) { if (CharacterFunctions::toLowerCase (path1[i]) != CharacterFunctions::toLowerCase (path2[i])) break; ++commonBitLength; } return path1.substring (0, commonBitLength).removeCharacters ("/:").isNotEmpty(); } //============================================================================== bool isJuceFolder (const File& folder) { return folder.getFileName().containsIgnoreCase ("juce") && folder.getChildFile ("modules").isDirectory(); } static File lookInFolderForJuceFolder (const File& folder) { for (DirectoryIterator di (folder, false, "*juce*", File::findDirectories); di.next();) { if (isJuceFolder (di.getFile())) return di.getFile(); } return File::nonexistent; } File findParentJuceFolder (const File& file) { File f (file); while (f.exists() && f.getParentDirectory() != f) { if (isJuceFolder (f)) return f; File found = lookInFolderForJuceFolder (f); if (found.exists()) return found; f = f.getParentDirectory(); } return File::nonexistent; } File findDefaultJuceFolder() { File f = findParentJuceFolder (File::getSpecialLocation (File::currentApplicationFile)); if (! f.exists()) f = lookInFolderForJuceFolder (File::getSpecialLocation (File::userHomeDirectory)); if (! f.exists()) f = lookInFolderForJuceFolder (File::getSpecialLocation (File::userDocumentsDirectory)); return f; } }