/* * SPDX-FileCopyrightText: 2011 Peter Penz * SPDX-FileCopyrightText: 2013 Frank Reininghaus * * SPDX-License-Identifier: GPL-2.0-or-later */ #include "kdirectorycontentscounterworker.h" // Required includes for subItemsCount(): #ifdef Q_OS_WIN #include #else #include #include #endif #include "dolphin_detailsmodesettings.h" KDirectoryContentsCounterWorker::KDirectoryContentsCounterWorker(QObject *parent) : QObject(parent) { qRegisterMetaType(); } #ifndef Q_OS_WIN KDirectoryContentsCounterWorker::CountResult walkDir(const QString &dirPath, const bool countHiddenFiles, const bool countDirectoriesOnly, const uint allowedRecursiveLevel) { int count = -1; long size = -1; auto dir = QT_OPENDIR(QFile::encodeName(dirPath)); if (dir) { count = 0; size = 0; QT_DIRENT *dirEntry; QT_STATBUF buf; while ((dirEntry = QT_READDIR(dir))) { if (dirEntry->d_name[0] == '.') { if (dirEntry->d_name[1] == '\0' || !countHiddenFiles) { // Skip "." or hidden files continue; } if (dirEntry->d_name[1] == '.' && dirEntry->d_name[2] == '\0') { // Skip ".." continue; } } // If only directories are counted, consider an unknown file type and links also // as directory instead of trying to do an expensive stat() // (see bugs 292642 and 299997). const bool countEntry = !countDirectoriesOnly || dirEntry->d_type == DT_DIR || dirEntry->d_type == DT_LNK || dirEntry->d_type == DT_UNKNOWN; if (countEntry) { ++count; } if (allowedRecursiveLevel > 0) { QString nameBuf = QStringLiteral("%1/%2").arg(dirPath, dirEntry->d_name); if (dirEntry->d_type == DT_REG) { if (QT_STAT(nameBuf.toLocal8Bit(), &buf) == 0) { size += buf.st_size; } } if (dirEntry->d_type == DT_DIR) { // recursion for dirs auto subdirResult = walkDir(nameBuf, countHiddenFiles, countDirectoriesOnly, allowedRecursiveLevel - 1); if (subdirResult.size > 0) { size += subdirResult.size; } } } } QT_CLOSEDIR(dir); } return KDirectoryContentsCounterWorker::CountResult{count, size}; } #endif KDirectoryContentsCounterWorker::CountResult KDirectoryContentsCounterWorker::subItemsCount(const QString &path, Options options) { const bool countHiddenFiles = options & CountHiddenFiles; const bool countDirectoriesOnly = options & CountDirectoriesOnly; #ifdef Q_OS_WIN QDir dir(path); QDir::Filters filters = QDir::NoDotAndDotDot | QDir::System; if (countHiddenFiles) { filters |= QDir::Hidden; } if (countDirectoriesOnly) { filters |= QDir::Dirs; } else { filters |= QDir::AllEntries; } return {static_cast(dir.entryList(filters).count()), 0}; #else const uint maxRecursiveLevel = DetailsModeSettings::directorySizeCount() ? 1 : DetailsModeSettings::recursiveDirectorySizeLimit(); auto res = walkDir(path, countHiddenFiles, countDirectoriesOnly, maxRecursiveLevel); return res; #endif } void KDirectoryContentsCounterWorker::countDirectoryContents(const QString &path, Options options) { auto res = subItemsCount(path, options); Q_EMIT result(path, res.count, res.size); }