2015-05-06 14:53:27 +03:00
|
|
|
/*
|
|
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
|
|
* Copyright (C) 2012 Christophe Dumez
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* In addition, as a special exception, the copyright holders give permission to
|
|
|
|
* link this program with the OpenSSL project's "OpenSSL" library (or with
|
|
|
|
* modified versions of it that use the same license as the "OpenSSL" library),
|
|
|
|
* and distribute the linked executables. You must obey the GNU General Public
|
|
|
|
* License in all respects for all of the code used other than "OpenSSL". If you
|
|
|
|
* modify file(s), you may extend this exception to your version of the file(s),
|
|
|
|
* but you are not obligated to do so. If you do not wish to do so, delete this
|
|
|
|
* exception statement from your version.
|
|
|
|
*
|
|
|
|
* Contact : chris@qbittorrent.org
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QSettings>
|
2015-08-09 15:27:56 +08:00
|
|
|
#include <QDirIterator>
|
2015-05-06 14:53:27 +03:00
|
|
|
#include <QCoreApplication>
|
|
|
|
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
#include <CoreServices/CoreServices.h>
|
|
|
|
#include <Carbon/Carbon.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef Q_OS_WIN
|
|
|
|
#if defined(Q_OS_MAC) || defined(Q_OS_FREEBSD)
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/mount.h>
|
|
|
|
#elif defined(Q_OS_HAIKU)
|
|
|
|
#include <kernel/fs_info.h>
|
|
|
|
#else
|
|
|
|
#include <sys/vfs.h>
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#include <shlobj.h>
|
|
|
|
#include <winbase.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if (QT_VERSION < QT_VERSION_CHECK(5, 0, 0))
|
2015-10-07 20:46:06 +03:00
|
|
|
|
|
|
|
#ifndef DISABLE_GUI
|
2015-05-06 14:53:27 +03:00
|
|
|
#include <QDesktopServices>
|
2015-10-07 20:46:06 +03:00
|
|
|
#endif
|
|
|
|
|
2015-05-06 14:53:27 +03:00
|
|
|
#else
|
|
|
|
#include <QStandardPaths>
|
|
|
|
#endif
|
2015-10-07 20:46:06 +03:00
|
|
|
|
2015-05-06 14:53:27 +03:00
|
|
|
|
|
|
|
#include "misc.h"
|
|
|
|
#include "fs.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a path to a string suitable for display.
|
|
|
|
* This function makes sure the directory separator used is consistent
|
|
|
|
* with the OS being run.
|
|
|
|
*/
|
|
|
|
QString Utils::Fs::toNativePath(const QString& path)
|
|
|
|
{
|
|
|
|
return QDir::toNativeSeparators(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::fromNativePath(const QString &path)
|
|
|
|
{
|
|
|
|
return QDir::fromNativeSeparators(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the file extension part of a file name.
|
|
|
|
*/
|
|
|
|
QString Utils::Fs::fileExtension(const QString &filename)
|
|
|
|
{
|
|
|
|
QString ext = QString(filename).remove(".!qB");
|
|
|
|
const int point_index = ext.lastIndexOf(".");
|
|
|
|
return (point_index >= 0) ? ext.mid(point_index + 1) : QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::fileName(const QString& file_path)
|
|
|
|
{
|
|
|
|
QString path = fromNativePath(file_path);
|
|
|
|
const int slash_index = path.lastIndexOf("/");
|
|
|
|
if (slash_index == -1)
|
|
|
|
return path;
|
|
|
|
return path.mid(slash_index + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::folderName(const QString& file_path)
|
|
|
|
{
|
|
|
|
QString path = fromNativePath(file_path);
|
|
|
|
const int slash_index = path.lastIndexOf("/");
|
|
|
|
if (slash_index == -1)
|
|
|
|
return path;
|
|
|
|
return path.left(slash_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-08-09 15:27:56 +08:00
|
|
|
* This function will first remove system cache files, e.g. `Thumbs.db`,
|
|
|
|
* `.DS_Store`. Then will try to remove the whole tree if the tree consist
|
|
|
|
* only of folders
|
2015-05-06 14:53:27 +03:00
|
|
|
*/
|
2015-08-09 15:27:56 +08:00
|
|
|
bool Utils::Fs::smartRemoveEmptyFolderTree(const QString& path)
|
2015-05-06 14:53:27 +03:00
|
|
|
{
|
2015-08-09 15:27:56 +08:00
|
|
|
if (!QDir(path).exists())
|
2015-05-06 14:53:27 +03:00
|
|
|
return false;
|
|
|
|
|
2015-08-09 15:27:56 +08:00
|
|
|
static const QStringList deleteFilesList = {
|
|
|
|
// Windows
|
|
|
|
"Thumbs.db",
|
|
|
|
"desktop.ini",
|
|
|
|
// Linux
|
|
|
|
".directory",
|
|
|
|
// Mac OS
|
|
|
|
".DS_Store"
|
|
|
|
};
|
|
|
|
|
|
|
|
// travel from the deepest folder and remove anything unwanted on the way out.
|
|
|
|
QStringList dirList(path + "/"); // get all sub directories paths
|
|
|
|
QDirIterator iter(path, (QDir::AllDirs | QDir::NoDotAndDotDot), QDirIterator::Subdirectories);
|
|
|
|
while (iter.hasNext())
|
|
|
|
dirList << iter.next() + "/";
|
|
|
|
std::sort(dirList.begin(), dirList.end(), [](const QString &l, const QString &r) { return l.count("/") > r.count("/"); }); // sort descending by directory depth
|
|
|
|
|
|
|
|
for (const QString &p : dirList) {
|
|
|
|
// remove unwanted files
|
|
|
|
for (const QString &f : deleteFilesList) {
|
|
|
|
forceRemove(p + f);
|
2015-05-06 14:53:27 +03:00
|
|
|
}
|
2015-08-09 15:27:56 +08:00
|
|
|
|
|
|
|
// remove temp files on linux (file ends with '~'), e.g. `filename~`
|
|
|
|
QDir dir(p);
|
|
|
|
QStringList tmpFileList = dir.entryList(QDir::Files);
|
|
|
|
for (const QString &f : tmpFileList) {
|
|
|
|
if (f.endsWith("~"))
|
|
|
|
forceRemove(p + f);
|
2015-05-06 14:53:27 +03:00
|
|
|
}
|
2015-08-09 15:27:56 +08:00
|
|
|
|
|
|
|
// remove directory if empty
|
|
|
|
dir.rmdir(p);
|
2015-05-06 14:53:27 +03:00
|
|
|
}
|
2015-08-09 15:27:56 +08:00
|
|
|
|
|
|
|
return QDir(path).exists();
|
2015-05-06 14:53:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the file with the given file_path.
|
|
|
|
*
|
|
|
|
* This function will try to fix the file permissions before removing it.
|
|
|
|
*/
|
|
|
|
bool Utils::Fs::forceRemove(const QString& file_path)
|
|
|
|
{
|
|
|
|
QFile f(file_path);
|
|
|
|
if (!f.exists())
|
|
|
|
return true;
|
|
|
|
// Make sure we have read/write permissions
|
|
|
|
f.setPermissions(f.permissions() | QFile::ReadOwner | QFile::WriteOwner | QFile::ReadUser | QFile::WriteUser);
|
|
|
|
// Remove the file
|
|
|
|
return f.remove();
|
|
|
|
}
|
|
|
|
|
2015-06-11 20:43:41 +03:00
|
|
|
/**
|
|
|
|
* Removes directory and its content recursively.
|
|
|
|
*
|
|
|
|
*/
|
2015-06-17 19:18:35 +03:00
|
|
|
void Utils::Fs::removeDirRecursive(const QString& dirName)
|
|
|
|
{
|
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0))
|
|
|
|
QDir(dirName).removeRecursively();
|
|
|
|
#else
|
2015-06-11 20:43:41 +03:00
|
|
|
QDir dir(dirName);
|
|
|
|
|
|
|
|
if (!dir.exists()) return;
|
|
|
|
|
|
|
|
Q_FOREACH(QFileInfo info, dir.entryInfoList(QDir::NoDotAndDotDot |
|
|
|
|
QDir::System |
|
|
|
|
QDir::Hidden |
|
|
|
|
QDir::AllDirs |
|
|
|
|
QDir::Files, QDir::DirsFirst)) {
|
|
|
|
if (info.isDir()) removeDirRecursive(info.absoluteFilePath());
|
|
|
|
else forceRemove(info.absoluteFilePath());
|
|
|
|
}
|
|
|
|
|
|
|
|
dir.rmdir(dirName);
|
2015-06-17 19:18:35 +03:00
|
|
|
#endif
|
2015-06-11 20:43:41 +03:00
|
|
|
}
|
|
|
|
|
2015-05-06 14:53:27 +03:00
|
|
|
/**
|
|
|
|
* Returns the size of a file.
|
|
|
|
* If the file is a folder, it will compute its size based on its content.
|
|
|
|
*
|
|
|
|
* Returns -1 in case of error.
|
|
|
|
*/
|
|
|
|
qint64 Utils::Fs::computePathSize(const QString& path)
|
|
|
|
{
|
|
|
|
// Check if it is a file
|
|
|
|
QFileInfo fi(path);
|
|
|
|
if (!fi.exists()) return -1;
|
|
|
|
if (fi.isFile()) return fi.size();
|
|
|
|
// Compute folder size based on its content
|
|
|
|
qint64 size = 0;
|
|
|
|
foreach (const QFileInfo &subfi, QDir(path).entryInfoList(QDir::Dirs|QDir::Files)) {
|
|
|
|
if (subfi.fileName().startsWith(".")) continue;
|
|
|
|
if (subfi.isDir())
|
|
|
|
size += computePathSize(subfi.absoluteFilePath());
|
|
|
|
else
|
|
|
|
size += subfi.size();
|
|
|
|
}
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes deep comparison of two files to make sure they are identical.
|
|
|
|
*/
|
|
|
|
bool Utils::Fs::sameFiles(const QString& path1, const QString& path2)
|
|
|
|
{
|
|
|
|
QFile f1(path1), f2(path2);
|
|
|
|
if (!f1.exists() || !f2.exists()) return false;
|
|
|
|
if (f1.size() != f2.size()) return false;
|
|
|
|
if (!f1.open(QIODevice::ReadOnly)) return false;
|
|
|
|
if (!f2.open(QIODevice::ReadOnly)) {
|
|
|
|
f1.close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
bool same = true;
|
|
|
|
while(!f1.atEnd() && !f2.atEnd()) {
|
|
|
|
if (f1.read(1024) != f2.read(1024)) {
|
|
|
|
same = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f1.close(); f2.close();
|
|
|
|
return same;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::toValidFileSystemName(QString filename)
|
|
|
|
{
|
|
|
|
qDebug("toValidFSName: %s", qPrintable(filename));
|
|
|
|
const QRegExp regex("[\\\\/:?\"*<>|]");
|
|
|
|
filename.replace(regex, " ");
|
|
|
|
qDebug("toValidFSName, result: %s", qPrintable(filename));
|
|
|
|
return filename.trimmed();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Utils::Fs::isValidFileSystemName(const QString& filename)
|
|
|
|
{
|
|
|
|
if (filename.isEmpty()) return false;
|
|
|
|
const QRegExp regex("[\\\\/:?\"*<>|]");
|
|
|
|
return !filename.contains(regex);
|
|
|
|
}
|
|
|
|
|
|
|
|
qlonglong Utils::Fs::freeDiskSpaceOnPath(QString path)
|
|
|
|
{
|
|
|
|
if (path.isEmpty()) return -1;
|
|
|
|
QDir dir_path(path);
|
|
|
|
if (!dir_path.exists()) {
|
|
|
|
QStringList parts = path.split("/");
|
|
|
|
while (parts.size() > 1 && !QDir(parts.join("/")).exists()) {
|
|
|
|
parts.removeLast();
|
|
|
|
}
|
|
|
|
dir_path = QDir(parts.join("/"));
|
|
|
|
if (!dir_path.exists()) return -1;
|
|
|
|
}
|
|
|
|
Q_ASSERT(dir_path.exists());
|
|
|
|
|
|
|
|
#ifndef Q_OS_WIN
|
|
|
|
unsigned long long available;
|
|
|
|
#ifdef Q_OS_HAIKU
|
|
|
|
const QString statfs_path = dir_path.path() + "/.";
|
|
|
|
dev_t device = dev_for_path (qPrintable(statfs_path));
|
|
|
|
if (device >= 0) {
|
|
|
|
fs_info info;
|
|
|
|
if (fs_stat_dev(device, &info) == B_OK) {
|
|
|
|
available = ((unsigned long long)(info.free_blocks * info.block_size));
|
|
|
|
return available;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
#else
|
|
|
|
struct statfs stats;
|
|
|
|
const QString statfs_path = dir_path.path() + "/.";
|
|
|
|
const int ret = statfs(qPrintable(statfs_path), &stats);
|
|
|
|
if (ret == 0) {
|
|
|
|
available = ((unsigned long long)stats.f_bavail)
|
|
|
|
* ((unsigned long long)stats.f_bsize);
|
|
|
|
return available;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
typedef BOOL (WINAPI *GetDiskFreeSpaceEx_t)(LPCTSTR,
|
|
|
|
PULARGE_INTEGER,
|
|
|
|
PULARGE_INTEGER,
|
|
|
|
PULARGE_INTEGER);
|
|
|
|
GetDiskFreeSpaceEx_t pGetDiskFreeSpaceEx =
|
|
|
|
(GetDiskFreeSpaceEx_t)::GetProcAddress(::GetModuleHandleW(L"kernel32.dll"), "GetDiskFreeSpaceExW");
|
|
|
|
if (pGetDiskFreeSpaceEx) {
|
|
|
|
ULARGE_INTEGER bytesFree, bytesTotal;
|
|
|
|
unsigned long long *ret;
|
|
|
|
if (pGetDiskFreeSpaceEx((LPCTSTR)(toNativePath(dir_path.path())).utf16(), &bytesFree, &bytesTotal, NULL)) {
|
|
|
|
ret = (unsigned long long*)&bytesFree;
|
|
|
|
return *ret;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::branchPath(const QString& file_path, QString* removed)
|
|
|
|
{
|
|
|
|
QString ret = fromNativePath(file_path);
|
|
|
|
if (ret.endsWith("/"))
|
|
|
|
ret.chop(1);
|
|
|
|
const int slashIndex = ret.lastIndexOf("/");
|
|
|
|
if (slashIndex >= 0) {
|
|
|
|
if (removed)
|
|
|
|
*removed = ret.mid(slashIndex + 1);
|
|
|
|
ret = ret.left(slashIndex);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Utils::Fs::sameFileNames(const QString &first, const QString &second)
|
|
|
|
{
|
|
|
|
#if defined(Q_OS_UNIX) || defined(Q_WS_QWS)
|
|
|
|
return QString::compare(first, second, Qt::CaseSensitive) == 0;
|
|
|
|
#else
|
|
|
|
return QString::compare(first, second, Qt::CaseInsensitive) == 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::expandPath(const QString &path)
|
|
|
|
{
|
|
|
|
QString ret = fromNativePath(path.trimmed());
|
|
|
|
if (ret.isEmpty())
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
return QDir::cleanPath(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::expandPathAbs(const QString& path)
|
|
|
|
{
|
|
|
|
QString ret = expandPath(path);
|
|
|
|
|
|
|
|
if (!QDir::isAbsolutePath(ret))
|
|
|
|
ret = QDir(ret).absolutePath();
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::QDesktopServicesDataLocation()
|
|
|
|
{
|
|
|
|
QString result;
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
LPWSTR path=new WCHAR[256];
|
|
|
|
if (SHGetSpecialFolderPath(0, path, CSIDL_LOCAL_APPDATA, FALSE))
|
|
|
|
result = fromNativePath(QString::fromWCharArray(path));
|
|
|
|
if (!QCoreApplication::applicationName().isEmpty())
|
|
|
|
result += QLatin1String("/") + qApp->applicationName();
|
|
|
|
#else
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
FSRef ref;
|
|
|
|
OSErr err = FSFindFolder(kUserDomain, kApplicationSupportFolderType, false, &ref);
|
|
|
|
if (err)
|
|
|
|
return QString();
|
|
|
|
QString path;
|
|
|
|
QByteArray ba(2048, 0);
|
|
|
|
if (FSRefMakePath(&ref, reinterpret_cast<UInt8 *>(ba.data()), ba.size()) == noErr)
|
|
|
|
result = QString::fromUtf8(ba).normalized(QString::NormalizationForm_C);
|
|
|
|
result += QLatin1Char('/') + qApp->applicationName();
|
|
|
|
#else
|
|
|
|
QString xdgDataHome = QLatin1String(qgetenv("XDG_DATA_HOME"));
|
|
|
|
if (xdgDataHome.isEmpty())
|
|
|
|
xdgDataHome = QDir::homePath() + QLatin1String("/.local/share");
|
|
|
|
xdgDataHome += QLatin1String("/data/")
|
|
|
|
+ qApp->applicationName();
|
|
|
|
result = xdgDataHome;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
if (!result.endsWith("/"))
|
|
|
|
result += "/";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::QDesktopServicesCacheLocation()
|
|
|
|
{
|
|
|
|
QString result;
|
|
|
|
#if defined(Q_OS_WIN) || defined(Q_OS_OS2)
|
|
|
|
result = QDesktopServicesDataLocation() + QLatin1String("cache");
|
|
|
|
#else
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
// http://developer.apple.com/documentation/Carbon/Reference/Folder_Manager/Reference/reference.html
|
|
|
|
FSRef ref;
|
|
|
|
OSErr err = FSFindFolder(kUserDomain, kCachedDataFolderType, false, &ref);
|
|
|
|
if (err)
|
|
|
|
return QString();
|
|
|
|
QByteArray ba(2048, 0);
|
|
|
|
if (FSRefMakePath(&ref, reinterpret_cast<UInt8 *>(ba.data()), ba.size()) == noErr)
|
|
|
|
result = QString::fromUtf8(ba).normalized(QString::NormalizationForm_C);
|
|
|
|
result += QLatin1Char('/') + qApp->applicationName();
|
|
|
|
#else
|
|
|
|
QString xdgCacheHome = QLatin1String(qgetenv("XDG_CACHE_HOME"));
|
|
|
|
if (xdgCacheHome.isEmpty())
|
|
|
|
xdgCacheHome = QDir::homePath() + QLatin1String("/.cache");
|
|
|
|
xdgCacheHome += QLatin1Char('/') + QCoreApplication::applicationName();
|
|
|
|
result = xdgCacheHome;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
if (!result.endsWith("/"))
|
|
|
|
result += "/";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::QDesktopServicesDownloadLocation()
|
|
|
|
{
|
2015-06-22 14:58:49 +08:00
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0))
|
|
|
|
#if defined(Q_OS_WIN)
|
|
|
|
if (QSysInfo::windowsVersion() <= QSysInfo::WV_XP) // Windows XP
|
|
|
|
return QDir(QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation)).absoluteFilePath(
|
|
|
|
QCoreApplication::translate("fsutils", "Downloads"));
|
|
|
|
#endif
|
|
|
|
return QStandardPaths::writableLocation(QStandardPaths::DownloadLocation);
|
|
|
|
#else
|
|
|
|
|
|
|
|
#if defined(Q_OS_OS2)
|
|
|
|
return QDir(QDesktopServices::storageLocation(QDesktopServices::DocumentsLocation)).absoluteFilePath(
|
|
|
|
QCoreApplication::translate("fsutils", "Downloads"));
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(Q_OS_WIN)
|
2015-05-06 14:53:27 +03:00
|
|
|
// as long as it stays WinXP like we do the same on OS/2
|
|
|
|
// TODO: Use IKnownFolderManager to get path of FOLDERID_Downloads
|
|
|
|
// instead of hardcoding "Downloads"
|
|
|
|
// Unfortunately, this would break compatibility with WinXP
|
2015-06-22 14:58:49 +08:00
|
|
|
if (QSysInfo::windowsVersion() <= QSysInfo::WV_XP) // Windows XP
|
|
|
|
return QDir(QDesktopServices::storageLocation(QDesktopServices::DocumentsLocation)).absoluteFilePath(
|
|
|
|
QCoreApplication::translate("fsutils", "Downloads"));
|
|
|
|
else
|
|
|
|
return QDir(QDesktopServices::storageLocation(QDesktopServices::HomeLocation)).absoluteFilePath("Downloads");
|
2015-05-06 14:53:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if (defined(Q_OS_UNIX) && !defined(Q_OS_MAC))
|
|
|
|
QString save_path;
|
|
|
|
// Default save path on Linux
|
|
|
|
QString config_path = QString::fromLocal8Bit(qgetenv("XDG_CONFIG_HOME").constData());
|
|
|
|
if (config_path.isEmpty())
|
|
|
|
config_path = QDir::home().absoluteFilePath(".config");
|
|
|
|
|
|
|
|
QString user_dirs_file = config_path + "/user-dirs.dirs";
|
|
|
|
if (QFile::exists(user_dirs_file)) {
|
|
|
|
QSettings settings(user_dirs_file, QSettings::IniFormat);
|
|
|
|
// We need to force UTF-8 encoding here since this is not
|
|
|
|
// the default for Ini files.
|
|
|
|
settings.setIniCodec("UTF-8");
|
|
|
|
QString xdg_download_dir = settings.value("XDG_DOWNLOAD_DIR").toString();
|
|
|
|
if (!xdg_download_dir.isEmpty()) {
|
|
|
|
// Resolve $HOME environment variables
|
|
|
|
xdg_download_dir.replace("$HOME", QDir::homePath());
|
|
|
|
save_path = xdg_download_dir;
|
|
|
|
qDebug() << Q_FUNC_INFO << "SUCCESS: Using XDG path for downloads: " << save_path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fallback
|
|
|
|
if (!save_path.isEmpty() && !QFile::exists(save_path)) {
|
|
|
|
QDir().mkpath(save_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (save_path.isEmpty() || !QFile::exists(save_path)) {
|
|
|
|
save_path = QDir::home().absoluteFilePath(QCoreApplication::translate("fsutils", "Downloads"));
|
|
|
|
qDebug() << Q_FUNC_INFO << "using" << save_path << "as fallback since the XDG detection did not work";
|
|
|
|
}
|
|
|
|
|
|
|
|
return save_path;
|
|
|
|
#endif
|
|
|
|
|
2015-06-22 14:58:49 +08:00
|
|
|
#if defined(Q_OS_MAC)
|
|
|
|
// TODO: How to support this on Mac OS?
|
2015-05-06 14:53:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Fallback
|
|
|
|
return QDir::home().absoluteFilePath(QCoreApplication::translate("fsutils", "Downloads"));
|
2015-06-22 14:58:49 +08:00
|
|
|
#endif
|
2015-05-06 14:53:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::searchEngineLocation()
|
|
|
|
{
|
|
|
|
QString folder = "nova";
|
|
|
|
if (Utils::Misc::pythonVersion() >= 3)
|
|
|
|
folder = "nova3";
|
|
|
|
const QString location = expandPathAbs(QDesktopServicesDataLocation() + folder);
|
|
|
|
QDir locationDir(location);
|
|
|
|
if (!locationDir.exists())
|
|
|
|
locationDir.mkpath(locationDir.absolutePath());
|
|
|
|
return location;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Utils::Fs::cacheLocation()
|
|
|
|
{
|
|
|
|
QString location = expandPathAbs(QDesktopServicesCacheLocation());
|
|
|
|
QDir locationDir(location);
|
|
|
|
if (!locationDir.exists())
|
|
|
|
locationDir.mkpath(locationDir.absolutePath());
|
|
|
|
return location;
|
|
|
|
}
|