/* ============================================================================== This file is part of the JUCE 6 technical preview. Copyright (c) 2020 - Raw Material Software Limited You may use this code under the terms of the GPL v3 (see www.gnu.org/licenses). For this technical preview, this file is not subject to commercial licensing. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE DISCLAIMED. ============================================================================== */ namespace juce { namespace build_tools { //============================================================================== /** Manipulates a cross-platform partial file path. (Needed because File is designed for absolute paths on the active OS) */ class RelativePath { public: //============================================================================== enum RootFolder { unknown, projectFolder, buildTargetFolder }; //============================================================================== RelativePath() : root (unknown) {} RelativePath (const String& relPath, const RootFolder rootType) : path (unixStylePath (relPath)), root (rootType) {} RelativePath (const File& file, const File& rootFolder, const RootFolder rootType) : path (unixStylePath (getRelativePathFrom (file, rootFolder))), root (rootType) {} RootFolder getRoot() const { return root; } String toUnixStyle() const { return unixStylePath (path); } String toWindowsStyle() const { return windowsStylePath (path); } String getFileName() const { return getFakeFile().getFileName(); } String getFileNameWithoutExtension() const { return getFakeFile().getFileNameWithoutExtension(); } String getFileExtension() const { return getFakeFile().getFileExtension(); } bool hasFileExtension (juce::StringRef extension) const { return getFakeFile().hasFileExtension (extension); } bool isAbsolute() const { return isAbsolutePath (path); } RelativePath withFileExtension (const String& extension) const { return RelativePath (path.upToLastOccurrenceOf (".", ! extension.startsWithChar ('.'), false) + extension, root); } RelativePath getParentDirectory() const { String p (path); if (path.endsWithChar ('/')) p = p.dropLastCharacters (1); return RelativePath (p.upToLastOccurrenceOf ("/", false, false), root); } RelativePath getChildFile (const String& subpath) const { if (isAbsolutePath (subpath)) return RelativePath (subpath, root); String p (toUnixStyle()); if (! p.endsWithChar ('/')) p << '/'; return RelativePath (p + subpath, root); } RelativePath rebased (const File& originalRoot, const File& newRoot, const RootFolder newRootType) const { if (isAbsolute()) return RelativePath (path, newRootType); return RelativePath (getRelativePathFrom (originalRoot.getChildFile (toUnixStyle()), newRoot), newRootType); } private: //============================================================================== String path; RootFolder root; File getFakeFile() const { #if JUCE_WINDOWS if (isAbsolutePath (path)) { // This is a hack to convert unix-style absolute paths into valid absolute Windows paths to avoid hitting // an assertion in File::parseAbsolutePath(). if (path.startsWithChar (L'/') || path.startsWithChar (L'$') || path.startsWithChar (L'~')) return File (String ("C:\\") + windowsStylePath (path.substring (1))); return File (path); } #endif // This method gets called very often, so we'll cache this directory. static const File currentWorkingDirectory (File::getCurrentWorkingDirectory()); return currentWorkingDirectory.getChildFile (path); } }; } }