Index: base/file_path.cc |
=================================================================== |
--- base/file_path.cc (revision 0) |
+++ base/file_path.cc (revision 0) |
@@ -0,0 +1,175 @@ |
+// Copyright (c) 2008 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "base/file_path.h" |
+ |
+#if defined(FILE_PATH_USES_WIN_SEPARATORS) |
+const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("\\/"); |
+#else // FILE_PATH_USES_WIN_SEPARATORS |
+const FilePath::CharType FilePath::kSeparators[] = FILE_PATH_LITERAL("/"); |
+#endif // FILE_PATH_USES_WIN_SEPARATORS |
+ |
+const FilePath::CharType FilePath::kCurrentDirectory[] = FILE_PATH_LITERAL("."); |
+const FilePath::CharType FilePath::kParentDirectory[] = FILE_PATH_LITERAL(".."); |
+ |
+// Returns true if |character| is in kSeparators. |
+static bool IsSeparator(FilePath::CharType character) { |
+ for (size_t i = 0; i < arraysize(FilePath::kSeparators) - 1; ++i) { |
+ if (character == FilePath::kSeparators[i]) { |
+ return true; |
+ } |
+ } |
+ |
+ return false; |
+} |
+ |
+// libgen's dirname and basename aren't guaranteed to be thread-safe and aren't |
+// guaranteed to not modify their input strings, and in fact are implmeneted |
+// differently in this regard on different platforms. Don't use them, but |
+// adhere to their behavior. |
+FilePath FilePath::DirName() const { |
+ FilePath new_path(path_); |
+ new_path.StripTrailingSeparators(); |
+ |
+ // The drive letter, if any, always needs to remain in the output. If there |
+ // is no drive letter, as will always be the case on platforms which do not |
+ // support drive letters, letter will be npos, or -1, so the comparisons and |
+ // resizes below using letter will still be valid. |
+ StringType::size_type letter = new_path.FindDriveLetter(); |
+ |
+ StringType::size_type last_separator = |
+ new_path.path_.find_last_of(kSeparators, StringType::npos, |
+ arraysize(kSeparators) - 1); |
+ if (last_separator == StringType::npos) { |
+ // path_ is in the current directory. |
+ new_path.path_.resize(letter + 1); |
+ } else if (last_separator == letter + 1) { |
+ // path_ is in the root directory. |
+ new_path.path_.resize(letter + 2); |
+ } else if (last_separator == letter + 2 && |
+ IsSeparator(new_path.path_[letter + 1])) { |
+ // path_ is in "//" (possibly with a drive letter); leave the double |
+ // separator intact indicating alternate root. |
+ new_path.path_.resize(letter + 3); |
+ } else if (last_separator != 0) { |
+ // path_ is somewhere else, trim the basename. |
+ new_path.path_.resize(last_separator); |
+ } |
+ |
+ new_path.StripTrailingSeparators(); |
+ if (!new_path.path_.length()) |
+ new_path.path_ = kCurrentDirectory; |
+ |
+ return new_path; |
+} |
+ |
+FilePath FilePath::BaseName() const { |
+ FilePath new_path(path_); |
+ new_path.StripTrailingSeparators(); |
+ |
+ // The drive letter, if any, is always stripped. |
+ StringType::size_type letter = new_path.FindDriveLetter(); |
+ if (letter != StringType::npos) { |
+ new_path.path_.erase(0, letter + 1); |
+ } |
+ |
+ // Keep everything after the final separator, but if the pathname is only |
+ // one character and it's a separator, leave it alone. |
+ StringType::size_type last_separator = |
+ new_path.path_.find_last_of(kSeparators, StringType::npos, |
+ arraysize(kSeparators) - 1); |
+ if (last_separator != StringType::npos && |
+ last_separator < new_path.path_.length() - 1) { |
+ new_path.path_.erase(0, last_separator + 1); |
+ } |
+ |
+ return new_path; |
+} |
+ |
+FilePath FilePath::Append(const FilePath::StringType& component) const { |
+ if (path_.compare(kCurrentDirectory) == 0) { |
+ // Append normally doesn't do any normalization, but as a special case, |
+ // when appending to kCurrentDirectory, just return a new path for the |
+ // component argument. Appending component to kCurrentDirectory would |
+ // serve no purpose other than needlessly lengthening the path, and |
+ // it's likely in practice to wind up with FilePath objects containing |
+ // only kCurrentDirectory when calling DirName on a single relative path |
+ // component. |
+ return FilePath(component); |
+ } |
+ |
+ FilePath new_path(path_); |
+ new_path.StripTrailingSeparators(); |
+ |
+ // Don't append a separator if the path is empty (indicating the current |
+ // directory) or if the path component is empty (indicating nothing to |
+ // append). |
+ if (component.length() > 0 && new_path.path_.length() > 0) { |
+ |
+ // Don't append a separator if the path still ends with a trailing |
+ // separator after stripping (indicating the root directory). |
+ if (!IsSeparator(new_path.path_[new_path.path_.length() - 1])) { |
+ |
+ // Don't append a separator if the path is just a drive letter. |
+ if (new_path.FindDriveLetter() + 1 != new_path.path_.length()) { |
+ new_path.path_.append(1, kSeparators[0]); |
+ } |
+ } |
+ } |
+ |
+ new_path.path_.append(component); |
+ return new_path; |
+} |
+ |
+FilePath::StringType::size_type FilePath::FindDriveLetter() const { |
+#if defined(FILE_PATH_USES_DRIVE_LETTERS) |
+ // This is dependent on an ASCII-based character set, but that's a |
+ // reasonable assumption. iswalpha can be too inclusive here. |
+ if (path_.length() >= 2 && path_[1] == L':' && |
+ ((path_[0] >= L'A' && path_[0] <= L'Z') || |
+ (path_[0] >= L'a' && path_[0] <= L'z'))) { |
+ return 1; |
+ } |
+ return StringType::npos; |
+#else // FILE_PATH_USES_DRIVE_LETTERS |
+ return StringType::npos; |
+#endif // FILE_PATH_USES_DRIVE_LETTERS |
+} |
+ |
+bool FilePath::IsAbsolute() const { |
+#if defined(FILE_PATH_USES_DRIVE_LETTERS) |
+ StringType::size_type letter = FindDriveLetter(); |
+ if (letter != StringType::npos) { |
+ // Look for a separator right after the drive specification. |
+ return path_.length() > letter + 1 && IsSeparator(path_[letter + 1]); |
+ } |
+ // Look for a pair of leading separators. |
+ return path_.length() > 1 && IsSeparator(path_[0]) && IsSeparator(path_[1]); |
+#else // FILE_PATH_USES_DRIVE_LETTERS |
+ // Look for a separator in the first position. |
+ return path_.length() > 0 && IsSeparator(path_[0]); |
+#endif // FILE_PATH_USES_DRIVE_LETTERS |
+} |
+ |
+void FilePath::StripTrailingSeparators() { |
+ // If there is no drive letter, start will be 1, which will prevent stripping |
+ // the leading separator if there is only one separator. If there is a drive |
+ // letter, start will be set appropriately to prevent stripping the first |
+ // separator following the drive letter, if a separator immediately follows |
+ // the drive letter. |
+ StringType::size_type start = FindDriveLetter() + 2; |
+ |
+ StringType::size_type last_stripped = StringType::npos; |
+ for (StringType::size_type pos = path_.length(); |
+ pos > start && IsSeparator(path_[pos - 1]); |
+ --pos) { |
+ // If the string only has two separators and they're at the beginning, |
+ // don't strip them, unless the string began with more than two separators. |
+ if (pos != start + 1 || last_stripped == start + 2 || |
+ !IsSeparator(path_[start - 1])) { |
+ path_.resize(pos - 1); |
+ last_stripped = pos; |
+ } |
+ } |
+} |
Property changes on: base/file_path.cc |
___________________________________________________________________ |
Added: svn:eol-style |
+ LF |