/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #pragma once #include "utils_global.h" #include "filepath.h" #include #include #ifdef QT_WIDGETS_LIB #include #endif #include #include QT_BEGIN_NAMESPACE class QDataStream; class QTextStream; class QWidget; class QXmlStreamWriter; // for withNtfsPermissions #ifdef Q_OS_WIN extern Q_CORE_EXPORT int qt_ntfs_permission_lookup; #endif QT_END_NAMESPACE namespace Utils { class QTCREATOR_UTILS_EXPORT FileUtils { public: #ifdef QT_GUI_LIB class QTCREATOR_UTILS_EXPORT CopyAskingForOverwrite { public: CopyAskingForOverwrite(QWidget *dialogParent, const std::function &postOperation = {}); bool operator()(const FilePath &src, const FilePath &dest, QString *error); FilePaths files() const; private: QWidget *m_parent; FilePaths m_files; std::function m_postOperation; bool m_overwriteAll = false; bool m_skipAll = false; }; #endif // QT_GUI_LIB static bool copyRecursively(const FilePath &srcFilePath, const FilePath &tgtFilePath, QString *error = nullptr); template static bool copyRecursively(const FilePath &srcFilePath, const FilePath &tgtFilePath, QString *error, T &©Helper); static bool copyIfDifferent(const FilePath &srcFilePath, const FilePath &tgtFilePath); static QString fileSystemFriendlyName(const QString &name); static int indexOfQmakeUnfriendly(const QString &name, int startpos = 0); static QString qmakeFriendlyName(const QString &name); static bool makeWritable(const FilePath &path); static QString normalizedPathName(const QString &name); static bool isRelativePath(const QString &fileName); static bool isAbsolutePath(const QString &fileName) { return !isRelativePath(fileName); } static FilePath commonPath(const FilePath &oldCommonPath, const FilePath &fileName); static FilePath commonPath(const FilePaths &paths); static QByteArray fileId(const FilePath &fileName); static FilePath homePath(); static FilePaths toFilePathList(const QStringList &paths); static void iterateLsOutput(const FilePath &base, const QStringList &entries, const FileFilter &filter, const std::function &callBack); #ifdef QT_WIDGETS_LIB static void setDialogParentGetter(const std::function &getter); static FilePath getOpenFilePath(QWidget *parent, const QString &caption, const FilePath &dir = {}, const QString &filter = {}, QString *selectedFilter = nullptr, QFileDialog::Options options = {}, bool fromDeviceIfShiftIsPressed = false); static FilePath getSaveFilePath(QWidget *parent, const QString &caption, const FilePath &dir = {}, const QString &filter = {}, QString *selectedFilter = nullptr, QFileDialog::Options options = {}); static FilePath getExistingDirectory(QWidget *parent, const QString &caption, const FilePath &dir = {}, QFileDialog::Options options = QFileDialog::ShowDirsOnly); static FilePaths getOpenFilePaths(QWidget *parent, const QString &caption, const FilePath &dir = {}, const QString &filter = {}, QString *selectedFilter = nullptr, QFileDialog::Options options = {}); static FilePath getOpenFilePathFromDevice(QWidget *parent, const QString &caption, const FilePath &dir = {}, const QString &filter = {}, QString *selectedFilter = nullptr, QFileDialog::Options options = {}); #endif }; template bool FileUtils::copyRecursively(const FilePath &srcFilePath, const FilePath &tgtFilePath, QString *error, T &©Helper) { if (srcFilePath.isDir()) { if (!tgtFilePath.exists()) { if (!tgtFilePath.ensureWritableDir()) { if (error) { *error = QCoreApplication::translate("Utils::FileUtils", "Failed to create directory \"%1\".") .arg(tgtFilePath.toUserOutput()); } return false; } } const QDir sourceDir(srcFilePath.toString()); const QStringList fileNames = sourceDir.entryList( QDir::Files | QDir::Dirs | QDir::NoDotAndDotDot | QDir::Hidden | QDir::System); for (const QString &fileName : fileNames) { const FilePath newSrcFilePath = srcFilePath / fileName; const FilePath newTgtFilePath = tgtFilePath / fileName; if (!copyRecursively(newSrcFilePath, newTgtFilePath, error, copyHelper)) return false; } } else { if (!copyHelper(srcFilePath, tgtFilePath, error)) return false; } return true; } // for actually finding out if e.g. directories are writable on Windows #ifdef Q_OS_WIN template T withNtfsPermissions(const std::function &task) { qt_ntfs_permission_lookup++; T result = task(); qt_ntfs_permission_lookup--; return result; } template <> QTCREATOR_UTILS_EXPORT void withNtfsPermissions(const std::function &task); #else // Q_OS_WIN template T withNtfsPermissions(const std::function &task) { return task(); } #endif // Q_OS_WIN class QTCREATOR_UTILS_EXPORT FileReader { Q_DECLARE_TR_FUNCTIONS(Utils::FileUtils) // sic! public: static QByteArray fetchQrc(const QString &fileName); // Only for internal resources bool fetch(const FilePath &filePath, QIODevice::OpenMode mode = QIODevice::NotOpen); // QIODevice::ReadOnly is implicit bool fetch(const FilePath &filePath, QIODevice::OpenMode mode, QString *errorString); bool fetch(const FilePath &filePath, QString *errorString) { return fetch(filePath, QIODevice::NotOpen, errorString); } #ifdef QT_GUI_LIB bool fetch(const FilePath &filePath, QIODevice::OpenMode mode, QWidget *parent); bool fetch(const FilePath &filePath, QWidget *parent) { return fetch(filePath, QIODevice::NotOpen, parent); } #endif // QT_GUI_LIB const QByteArray &data() const { return m_data; } const QString &errorString() const { return m_errorString; } private: QByteArray m_data; QString m_errorString; }; class QTCREATOR_UTILS_EXPORT FileSaverBase { Q_DECLARE_TR_FUNCTIONS(Utils::FileUtils) // sic! public: FileSaverBase(); virtual ~FileSaverBase(); FilePath filePath() const { return m_filePath; } bool hasError() const { return m_hasError; } QString errorString() const { return m_errorString; } virtual bool finalize(); bool finalize(QString *errStr); #ifdef QT_GUI_LIB bool finalize(QWidget *parent); #endif bool write(const char *data, int len); bool write(const QByteArray &bytes); bool setResult(QTextStream *stream); bool setResult(QDataStream *stream); bool setResult(QXmlStreamWriter *stream); bool setResult(bool ok); QFile *file() { return m_file.get(); } protected: std::unique_ptr m_file; FilePath m_filePath; QString m_errorString; bool m_hasError = false; private: Q_DISABLE_COPY(FileSaverBase) }; class QTCREATOR_UTILS_EXPORT FileSaver : public FileSaverBase { Q_DECLARE_TR_FUNCTIONS(Utils::FileUtils) // sic! public: // QIODevice::WriteOnly is implicit explicit FileSaver(const FilePath &filePath, QIODevice::OpenMode mode = QIODevice::NotOpen); bool finalize() override; using FileSaverBase::finalize; private: bool m_isSafe = false; }; class QTCREATOR_UTILS_EXPORT TempFileSaver : public FileSaverBase { Q_DECLARE_TR_FUNCTIONS(Utils::FileUtils) // sic! public: explicit TempFileSaver(const QString &templ = QString()); ~TempFileSaver() override; void setAutoRemove(bool on) { m_autoRemove = on; } private: bool m_autoRemove = true; }; QTCREATOR_UTILS_EXPORT QTextStream &operator<<(QTextStream &s, const FilePath &fn); bool isRelativePathHelper(const QString &path, OsType osType); QString doCleanPath(const QString &input); } // namespace Utils