2010-05-10 23:50:31 +02:00
|
|
|
/* This file is part of Clementine.
|
2014-11-02 19:36:21 +01:00
|
|
|
Copyright 2010-2014, David Sansome <me@davidsansome.com>
|
|
|
|
Copyright 2010-2012, 2014, John Maguire <john.maguire@gmail.com>
|
|
|
|
Copyright 2011, 2014, Arnaud Bienner <arnaud.bienner@gmail.com>
|
|
|
|
Copyright 2012, Alan Briolat <alan.briolat@gmail.com>
|
|
|
|
Copyright 2012, Veniamin Gvozdikov <G.Veniamin@gmail.com>
|
|
|
|
Copyright 2013-2014, Andreas <asfa194@gmail.com>
|
|
|
|
Copyright 2013, Glad Olus <gladolus@gmx.com>
|
|
|
|
Copyright 2013, graehl <graehl@gmail.com>
|
|
|
|
Copyright 2014, vkrishtal <krishtalhost@gmail.com>
|
|
|
|
Copyright 2014, Krzysztof Sobiecki <sobkas@gmail.com>
|
2010-05-10 23:50:31 +02:00
|
|
|
|
|
|
|
Clementine 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 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Clementine 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 Clementine. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "utilities.h"
|
|
|
|
|
2012-11-06 15:38:15 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2011-10-15 21:48:48 +02:00
|
|
|
#include <QApplication>
|
2010-10-17 22:53:15 +02:00
|
|
|
#include <QDateTime>
|
2011-03-17 20:52:21 +01:00
|
|
|
#include <QDesktopServices>
|
2010-07-30 00:58:23 +02:00
|
|
|
#include <QDir>
|
2013-12-31 15:21:50 +01:00
|
|
|
#include <QFile>
|
2010-08-10 21:42:43 +02:00
|
|
|
#include <QIODevice>
|
2012-11-06 15:38:15 +01:00
|
|
|
#include <QMetaEnum>
|
2011-10-15 21:48:48 +02:00
|
|
|
#include <QMouseEvent>
|
2010-06-12 23:20:53 +02:00
|
|
|
#include <QStringList>
|
2011-11-28 19:11:09 +01:00
|
|
|
#include <QTcpServer>
|
2010-08-01 16:13:27 +02:00
|
|
|
#include <QTemporaryFile>
|
2011-03-17 20:52:21 +01:00
|
|
|
#include <QUrl>
|
2021-02-01 07:32:38 +01:00
|
|
|
#include <QUrlQuery>
|
2011-10-15 21:48:48 +02:00
|
|
|
#include <QWidget>
|
2012-01-07 22:51:02 +01:00
|
|
|
#include <QXmlStreamReader>
|
2020-09-18 16:15:19 +02:00
|
|
|
#include <QtDebug>
|
|
|
|
#include <QtGlobal>
|
|
|
|
#include <memory>
|
2012-11-06 15:38:15 +01:00
|
|
|
|
2020-09-18 16:15:19 +02:00
|
|
|
#include "config.h"
|
2013-09-29 10:43:34 +02:00
|
|
|
#include "core/application.h"
|
2012-11-06 15:38:15 +01:00
|
|
|
#include "core/logging.h"
|
2021-03-17 05:23:44 +01:00
|
|
|
#include "core/timeconstants.h"
|
2012-11-06 15:38:15 +01:00
|
|
|
|
2021-04-09 07:58:26 +02:00
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
|
|
|
|
#include <QRandomGenerator>
|
|
|
|
#endif
|
|
|
|
|
2010-07-30 00:58:23 +02:00
|
|
|
#if defined(Q_OS_UNIX)
|
2014-02-07 16:34:20 +01:00
|
|
|
#include <sys/statvfs.h>
|
2010-07-30 00:58:23 +02:00
|
|
|
#elif defined(Q_OS_WIN32)
|
2014-02-07 16:34:20 +01:00
|
|
|
#include <windows.h>
|
2020-09-18 16:15:19 +02:00
|
|
|
|
2015-10-11 22:56:42 +02:00
|
|
|
#include <QProcess>
|
2010-07-30 00:58:23 +02:00
|
|
|
#endif
|
2010-07-30 00:16:12 +02:00
|
|
|
|
2012-02-26 16:05:46 +01:00
|
|
|
#ifdef Q_OS_LINUX
|
2014-02-07 16:34:20 +01:00
|
|
|
#include <sys/syscall.h>
|
2020-09-18 16:15:19 +02:00
|
|
|
#include <unistd.h>
|
2012-02-26 16:05:46 +01:00
|
|
|
#endif
|
|
|
|
#ifdef Q_OS_DARWIN
|
2014-02-07 16:34:20 +01:00
|
|
|
#include <sys/resource.h>
|
2012-02-26 16:05:46 +01:00
|
|
|
#endif
|
|
|
|
|
2010-12-09 13:34:08 +01:00
|
|
|
#ifdef Q_OS_DARWIN
|
2020-09-18 16:15:19 +02:00
|
|
|
#include <QProcess>
|
|
|
|
|
|
|
|
#include "CoreServices/CoreServices.h"
|
|
|
|
#include "IOKit/ps/IOPSKeys.h"
|
|
|
|
#include "IOKit/ps/IOPowerSources.h"
|
2014-02-07 16:34:20 +01:00
|
|
|
#include "core/mac_startup.h"
|
|
|
|
#include "core/mac_utilities.h"
|
|
|
|
#include "core/scoped_cftyperef.h"
|
2010-12-09 13:34:08 +01:00
|
|
|
#endif
|
|
|
|
|
2010-05-10 23:50:31 +02:00
|
|
|
namespace Utilities {
|
|
|
|
|
2010-07-03 23:05:55 +02:00
|
|
|
static QString tr(const char* str) {
|
|
|
|
return QCoreApplication::translate("", str);
|
|
|
|
}
|
|
|
|
|
2011-02-21 21:59:30 +01:00
|
|
|
QString PrettyTimeDelta(int seconds) {
|
|
|
|
return (seconds >= 0 ? "+" : "-") + PrettyTime(seconds);
|
|
|
|
}
|
|
|
|
|
2018-11-17 14:29:16 +01:00
|
|
|
QString PrettyTime(int seconds, bool always_show_hours) {
|
2010-05-10 23:50:31 +02:00
|
|
|
// last.fm sometimes gets the track length wrong, so you end up with
|
|
|
|
// negative times.
|
|
|
|
seconds = qAbs(seconds);
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
int hours = seconds / (60 * 60);
|
2010-05-10 23:50:31 +02:00
|
|
|
int minutes = (seconds / 60) % 60;
|
|
|
|
seconds %= 60;
|
|
|
|
|
|
|
|
QString ret;
|
2018-11-17 14:29:16 +01:00
|
|
|
if (hours || always_show_hours)
|
2020-01-05 00:12:32 +01:00
|
|
|
ret = QString::asprintf("%d:%02d:%02d", hours, minutes,
|
2020-01-05 00:17:18 +01:00
|
|
|
seconds); // NOLINT(runtime/printf)
|
2010-05-10 23:50:31 +02:00
|
|
|
else
|
2020-01-05 00:17:18 +01:00
|
|
|
ret = QString::asprintf("%d:%02d", minutes,
|
|
|
|
seconds); // NOLINT(runtime/printf)
|
2010-05-10 23:50:31 +02:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-11-17 14:29:16 +01:00
|
|
|
QString PrettyTimeNanosec(qint64 nanoseconds, bool always_show_hours) {
|
|
|
|
return PrettyTime(nanoseconds / kNsecPerSec, always_show_hours);
|
2011-02-13 19:34:30 +01:00
|
|
|
}
|
|
|
|
|
2010-06-12 23:20:53 +02:00
|
|
|
QString WordyTime(quint64 seconds) {
|
2017-12-13 19:37:34 +01:00
|
|
|
quint64 days = seconds / (kSecsPerDay);
|
|
|
|
quint64 remaining_hours = (seconds - days * kSecsPerDay) / (60 * 60);
|
2010-06-12 23:20:53 +02:00
|
|
|
|
2014-11-01 19:26:05 +01:00
|
|
|
// TODO(David Sansome): Make the plural rules translatable
|
2010-06-12 23:20:53 +02:00
|
|
|
QStringList parts;
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
if (days) parts << (days == 1 ? tr("1 day") : tr("%1 days").arg(days));
|
2017-12-13 19:37:34 +01:00
|
|
|
|
|
|
|
// Since PrettyTime does not return the hour if it is 0, we need to add it
|
|
|
|
// explicitly for durations longer than 1 day.
|
|
|
|
parts << (days && !remaining_hours ? QString("0:") : QString()) +
|
|
|
|
PrettyTime(seconds - days * kSecsPerDay);
|
2010-06-12 23:20:53 +02:00
|
|
|
|
|
|
|
return parts.join(" ");
|
|
|
|
}
|
|
|
|
|
2011-02-13 19:34:30 +01:00
|
|
|
QString WordyTimeNanosec(qint64 nanoseconds) {
|
2011-02-14 20:34:37 +01:00
|
|
|
return WordyTime(nanoseconds / kNsecPerSec);
|
2011-02-13 19:34:30 +01:00
|
|
|
}
|
|
|
|
|
2010-10-17 22:53:15 +02:00
|
|
|
QString Ago(int seconds_since_epoch, const QLocale& locale) {
|
|
|
|
const QDateTime now = QDateTime::currentDateTime();
|
|
|
|
const QDateTime then = QDateTime::fromTime_t(seconds_since_epoch);
|
|
|
|
const int days_ago = then.date().daysTo(now.date());
|
2014-02-07 16:34:20 +01:00
|
|
|
const QString time =
|
|
|
|
then.time().toString(locale.timeFormat(QLocale::ShortFormat));
|
2010-10-17 22:53:15 +02:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
if (days_ago == 0) return tr("Today") + " " + time;
|
|
|
|
if (days_ago == 1) return tr("Yesterday") + " " + time;
|
|
|
|
if (days_ago <= 7) return tr("%1 days ago").arg(days_ago);
|
2010-10-17 22:53:15 +02:00
|
|
|
|
2013-08-29 16:30:20 +02:00
|
|
|
return then.date().toString(locale.dateFormat(QLocale::ShortFormat));
|
2010-10-17 22:53:15 +02:00
|
|
|
}
|
|
|
|
|
2012-08-27 13:24:28 +02:00
|
|
|
QString PrettyFutureDate(const QDate& date) {
|
|
|
|
const QDate now = QDate::currentDate();
|
|
|
|
const int delta_days = now.daysTo(date);
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
if (delta_days < 0) return QString();
|
|
|
|
if (delta_days == 0) return tr("Today");
|
|
|
|
if (delta_days == 1) return tr("Tomorrow");
|
|
|
|
if (delta_days <= 7) return tr("In %1 days").arg(delta_days);
|
|
|
|
if (delta_days <= 14) return tr("Next week");
|
2012-08-27 13:24:28 +02:00
|
|
|
|
|
|
|
return tr("In %1 weeks").arg(delta_days / 7);
|
|
|
|
}
|
|
|
|
|
2010-07-03 23:05:55 +02:00
|
|
|
QString PrettySize(quint64 bytes) {
|
|
|
|
QString ret;
|
|
|
|
|
|
|
|
if (bytes > 0) {
|
|
|
|
if (bytes <= 1000)
|
|
|
|
ret = QString::number(bytes) + " bytes";
|
2014-02-07 16:34:20 +01:00
|
|
|
else if (bytes <= 1000 * 1000)
|
2020-01-05 00:17:18 +01:00
|
|
|
ret = QString::asprintf(
|
|
|
|
"%.1f KB",
|
|
|
|
static_cast<float>(bytes) / 1000); // NOLINT(runtime/printf)
|
2014-02-07 16:34:20 +01:00
|
|
|
else if (bytes <= 1000 * 1000 * 1000)
|
2020-01-05 00:17:18 +01:00
|
|
|
ret = QString::asprintf(
|
|
|
|
"%.1f MB",
|
|
|
|
static_cast<float>(bytes) / (1000 * 1000)); // NOLINT(runtime/printf)
|
2010-07-03 23:05:55 +02:00
|
|
|
else
|
2020-01-05 00:17:18 +01:00
|
|
|
ret = QString::asprintf(
|
|
|
|
"%.1f GB",
|
|
|
|
static_cast<float>(bytes) /
|
|
|
|
(1000 * 1000 * 1000)); // NOLINT(runtime/printf)
|
2010-07-03 23:05:55 +02:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-07-30 00:16:12 +02:00
|
|
|
quint64 FileSystemCapacity(const QString& path) {
|
2010-07-30 00:58:23 +02:00
|
|
|
#if defined(Q_OS_UNIX)
|
2010-07-30 00:16:12 +02:00
|
|
|
struct statvfs fs_info;
|
|
|
|
if (statvfs(path.toLocal8Bit().constData(), &fs_info) == 0)
|
2010-08-07 12:00:20 +02:00
|
|
|
return quint64(fs_info.f_blocks) * quint64(fs_info.f_bsize);
|
2010-07-30 00:58:23 +02:00
|
|
|
#elif defined(Q_OS_WIN32)
|
|
|
|
_ULARGE_INTEGER ret;
|
2014-02-07 16:34:20 +01:00
|
|
|
if (GetDiskFreeSpaceEx(
|
|
|
|
QDir::toNativeSeparators(path).toLocal8Bit().constData(), nullptr,
|
|
|
|
&ret, nullptr) != 0)
|
2010-07-30 00:58:23 +02:00
|
|
|
return ret.QuadPart;
|
|
|
|
#endif
|
2010-07-30 00:16:12 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
quint64 FileSystemFreeSpace(const QString& path) {
|
2010-07-30 00:58:23 +02:00
|
|
|
#if defined(Q_OS_UNIX)
|
2010-07-30 00:16:12 +02:00
|
|
|
struct statvfs fs_info;
|
|
|
|
if (statvfs(path.toLocal8Bit().constData(), &fs_info) == 0)
|
2010-08-07 12:00:20 +02:00
|
|
|
return quint64(fs_info.f_bavail) * quint64(fs_info.f_bsize);
|
2010-07-30 00:58:23 +02:00
|
|
|
#elif defined(Q_OS_WIN32)
|
|
|
|
_ULARGE_INTEGER ret;
|
2014-02-07 16:34:20 +01:00
|
|
|
if (GetDiskFreeSpaceEx(
|
|
|
|
QDir::toNativeSeparators(path).toLocal8Bit().constData(), &ret,
|
|
|
|
nullptr, nullptr) != 0)
|
2010-07-30 00:58:23 +02:00
|
|
|
return ret.QuadPart;
|
|
|
|
#endif
|
2010-07-30 00:16:12 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-01-28 22:43:10 +01:00
|
|
|
QString MakeTempDir(const QString template_name) {
|
2010-08-01 16:13:27 +02:00
|
|
|
QString path;
|
|
|
|
{
|
|
|
|
QTemporaryFile tempfile;
|
2014-02-07 16:34:20 +01:00
|
|
|
if (!template_name.isEmpty()) tempfile.setFileTemplate(template_name);
|
2011-01-28 22:43:10 +01:00
|
|
|
|
2010-08-01 16:13:27 +02:00
|
|
|
tempfile.open();
|
|
|
|
path = tempfile.fileName();
|
|
|
|
}
|
|
|
|
|
|
|
|
QDir d;
|
|
|
|
d.mkdir(path);
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2013-08-01 19:13:43 +02:00
|
|
|
QString GetTemporaryFileName() {
|
2013-08-01 18:13:14 +02:00
|
|
|
QString file;
|
|
|
|
{
|
|
|
|
QTemporaryFile tempfile;
|
2014-11-13 22:31:49 +01:00
|
|
|
// Do not delete the file, we want to do something with it
|
|
|
|
tempfile.setAutoRemove(false);
|
2013-08-01 18:13:14 +02:00
|
|
|
tempfile.open();
|
|
|
|
file = tempfile.fileName();
|
|
|
|
}
|
|
|
|
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2017-03-17 17:54:02 +01:00
|
|
|
QString SaveToTemporaryFile(const QByteArray& data) {
|
2017-03-17 19:05:01 +01:00
|
|
|
QTemporaryFile tempfile;
|
|
|
|
tempfile.setAutoRemove(false);
|
2017-03-17 17:54:02 +01:00
|
|
|
|
2017-03-17 19:05:01 +01:00
|
|
|
if (!tempfile.open()) {
|
2017-03-17 17:54:02 +01:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2017-03-17 19:05:01 +01:00
|
|
|
if (tempfile.write(data) != data.size()) {
|
|
|
|
tempfile.remove();
|
2017-03-17 17:54:02 +01:00
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2017-03-17 19:05:01 +01:00
|
|
|
tempfile.close();
|
|
|
|
return tempfile.fileName();
|
2017-03-17 17:54:02 +01:00
|
|
|
}
|
|
|
|
|
2014-04-25 06:57:31 +02:00
|
|
|
bool RemoveRecursive(const QString& path) {
|
2010-08-01 16:13:27 +02:00
|
|
|
QDir dir(path);
|
2014-02-10 14:29:07 +01:00
|
|
|
for (const QString& child :
|
2014-04-25 06:57:31 +02:00
|
|
|
dir.entryList(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Hidden)) {
|
2014-12-14 12:23:04 +01:00
|
|
|
if (!RemoveRecursive(path + "/" + child)) return false;
|
2014-04-25 06:57:31 +02:00
|
|
|
}
|
2010-08-01 16:13:27 +02:00
|
|
|
|
2014-02-10 14:29:07 +01:00
|
|
|
for (const QString& child :
|
2014-04-25 06:57:31 +02:00
|
|
|
dir.entryList(QDir::NoDotAndDotDot | QDir::Files | QDir::Hidden)) {
|
2014-12-14 12:23:04 +01:00
|
|
|
if (!QFile::remove(path + "/" + child)) return false;
|
2014-04-25 06:57:31 +02:00
|
|
|
}
|
|
|
|
|
2019-02-22 18:44:39 +01:00
|
|
|
return dir.rmdir(path);
|
2010-08-01 16:13:27 +02:00
|
|
|
}
|
|
|
|
|
2011-01-19 00:10:22 +01:00
|
|
|
bool CopyRecursive(const QString& source, const QString& destination) {
|
|
|
|
// Make the destination directory
|
|
|
|
QString dir_name = source.section('/', -1, -1);
|
|
|
|
QString dest_path = destination + "/" + dir_name;
|
|
|
|
QDir().mkpath(dest_path);
|
|
|
|
|
|
|
|
QDir dir(source);
|
2014-02-10 14:29:07 +01:00
|
|
|
for (const QString& child :
|
|
|
|
dir.entryList(QDir::NoDotAndDotDot | QDir::Dirs)) {
|
2011-01-19 00:10:22 +01:00
|
|
|
if (!CopyRecursive(source + "/" + child, dest_path)) {
|
2014-02-07 16:34:20 +01:00
|
|
|
qLog(Warning) << "Failed to copy dir" << source + "/" + child << "to"
|
|
|
|
<< dest_path;
|
2011-01-19 00:10:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-10 14:29:07 +01:00
|
|
|
for (const QString& child :
|
|
|
|
dir.entryList(QDir::NoDotAndDotDot | QDir::Files)) {
|
2011-01-19 00:10:22 +01:00
|
|
|
if (!QFile::copy(source + "/" + child, dest_path + "/" + child)) {
|
2014-02-07 16:34:20 +01:00
|
|
|
qLog(Warning) << "Failed to copy file" << source + "/" + child << "to"
|
|
|
|
<< dest_path;
|
2011-01-19 00:10:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-08-10 21:42:43 +02:00
|
|
|
bool Copy(QIODevice* source, QIODevice* destination) {
|
2014-02-07 16:34:20 +01:00
|
|
|
if (!source->open(QIODevice::ReadOnly)) return false;
|
2010-08-10 21:42:43 +02:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
if (!destination->open(QIODevice::WriteOnly)) return false;
|
2010-08-10 21:42:43 +02:00
|
|
|
|
|
|
|
const qint64 bytes = source->size();
|
2014-02-06 14:48:00 +01:00
|
|
|
std::unique_ptr<char[]> data(new char[bytes]);
|
2010-08-10 21:42:43 +02:00
|
|
|
qint64 pos = 0;
|
|
|
|
|
2010-08-12 18:35:43 +02:00
|
|
|
qint64 bytes_read;
|
|
|
|
do {
|
|
|
|
bytes_read = source->read(data.get() + pos, bytes - pos);
|
2014-02-07 16:34:20 +01:00
|
|
|
if (bytes_read == -1) return false;
|
2010-08-10 21:42:43 +02:00
|
|
|
|
|
|
|
pos += bytes_read;
|
2010-08-12 18:35:43 +02:00
|
|
|
} while (bytes_read > 0 && pos != bytes);
|
2010-08-10 21:42:43 +02:00
|
|
|
|
|
|
|
pos = 0;
|
2010-08-12 18:35:43 +02:00
|
|
|
qint64 bytes_written;
|
|
|
|
do {
|
|
|
|
bytes_written = destination->write(data.get() + pos, bytes - pos);
|
2014-02-07 16:34:20 +01:00
|
|
|
if (bytes_written == -1) return false;
|
2010-08-10 21:42:43 +02:00
|
|
|
|
|
|
|
pos += bytes_written;
|
2010-08-12 18:35:43 +02:00
|
|
|
} while (bytes_written > 0 && pos != bytes);
|
2010-08-10 21:42:43 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-10-25 01:46:05 +02:00
|
|
|
QString ColorToRgba(const QColor& c) {
|
|
|
|
return QString("rgba(%1, %2, %3, %4)")
|
2014-02-07 16:34:20 +01:00
|
|
|
.arg(c.red())
|
|
|
|
.arg(c.green())
|
|
|
|
.arg(c.blue())
|
|
|
|
.arg(c.alpha());
|
2010-10-25 01:46:05 +02:00
|
|
|
}
|
|
|
|
|
2010-12-09 13:34:08 +01:00
|
|
|
QString GetConfigPath(ConfigPath config) {
|
|
|
|
switch (config) {
|
2010-12-09 14:06:00 +01:00
|
|
|
case Path_Root: {
|
2020-02-19 06:57:29 +01:00
|
|
|
if (Application::IsPortable()) {
|
|
|
|
QDir d(QCoreApplication::applicationDirPath());
|
|
|
|
return d.filePath(Application::kPortableDataDir);
|
2013-09-29 10:43:34 +02:00
|
|
|
}
|
2014-02-07 16:34:20 +01:00
|
|
|
#ifdef Q_OS_DARWIN
|
|
|
|
return mac::GetApplicationSupportPath() + "/" +
|
|
|
|
QCoreApplication::organizationName();
|
|
|
|
#else
|
2019-12-10 18:45:06 +01:00
|
|
|
return QString("%1/%2").arg(
|
|
|
|
QStandardPaths::writableLocation(QStandardPaths::ConfigLocation),
|
|
|
|
QCoreApplication::organizationName());
|
2014-02-07 16:34:20 +01:00
|
|
|
#endif
|
|
|
|
} break;
|
2010-12-09 13:34:08 +01:00
|
|
|
|
2012-11-06 15:38:15 +01:00
|
|
|
case Path_CacheRoot: {
|
2013-09-29 10:43:34 +02:00
|
|
|
if (Application::kIsPortable) {
|
|
|
|
return GetConfigPath(Path_Root) + "/cache";
|
|
|
|
}
|
2014-02-07 16:34:20 +01:00
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_DARWIN)
|
2019-12-10 18:45:06 +01:00
|
|
|
return QString("%1/%2").arg(QStandardPaths::writableLocation(
|
|
|
|
QStandardPaths::GenericCacheLocation),
|
|
|
|
QCoreApplication::organizationName());
|
2014-02-07 16:34:20 +01:00
|
|
|
#else
|
|
|
|
return GetConfigPath(Path_Root);
|
|
|
|
#endif
|
|
|
|
} break;
|
2012-11-06 15:38:15 +01:00
|
|
|
|
2015-10-14 03:01:08 +02:00
|
|
|
case Path_Icons:
|
|
|
|
return GetConfigPath(Path_Root) + "/customiconset";
|
|
|
|
|
2010-12-09 14:06:00 +01:00
|
|
|
case Path_AlbumCovers:
|
|
|
|
return GetConfigPath(Path_Root) + "/albumcovers";
|
2010-12-09 13:34:08 +01:00
|
|
|
|
2010-12-09 14:06:00 +01:00
|
|
|
case Path_NetworkCache:
|
2012-11-06 15:38:15 +01:00
|
|
|
return GetConfigPath(Path_CacheRoot) + "/networkcache";
|
|
|
|
|
2012-05-25 18:18:07 +02:00
|
|
|
case Path_MoodbarCache:
|
2012-11-06 15:38:15 +01:00
|
|
|
return GetConfigPath(Path_CacheRoot) + "/moodbarcache";
|
2010-12-09 13:34:08 +01:00
|
|
|
|
2019-12-07 23:30:14 +01:00
|
|
|
case Path_PixmapCache:
|
|
|
|
return GetConfigPath(Path_CacheRoot) + "/pixmapcache";
|
|
|
|
|
2010-12-09 14:06:00 +01:00
|
|
|
case Path_GstreamerRegistry:
|
|
|
|
return GetConfigPath(Path_Root) +
|
2014-02-07 16:34:20 +01:00
|
|
|
QString("/gst-registry-%1-bin")
|
|
|
|
.arg(QCoreApplication::applicationVersion());
|
2010-12-09 13:34:08 +01:00
|
|
|
|
2010-12-14 16:00:46 +01:00
|
|
|
case Path_DefaultMusicLibrary:
|
2014-02-07 16:34:20 +01:00
|
|
|
#ifdef Q_OS_DARWIN
|
|
|
|
return mac::GetMusicDirectory();
|
|
|
|
#else
|
|
|
|
return QDir::homePath();
|
|
|
|
#endif
|
2010-12-14 16:00:46 +01:00
|
|
|
|
2011-04-29 21:44:51 +02:00
|
|
|
case Path_LocalSpotifyBlob:
|
|
|
|
return GetConfigPath(Path_Root) + "/spotifyblob";
|
|
|
|
|
2010-12-09 13:34:08 +01:00
|
|
|
default:
|
|
|
|
qFatal("%s", Q_FUNC_INFO);
|
2019-11-09 23:45:28 +01:00
|
|
|
return QString();
|
2010-12-09 13:34:08 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-09 13:59:25 +01:00
|
|
|
#ifdef Q_OS_DARWIN
|
2013-04-20 23:08:45 +02:00
|
|
|
// Better than openUrl(dirname(path)) - also highlights file at path
|
|
|
|
void RevealFileInFinder(QString const& path) {
|
|
|
|
QProcess::execute("/usr/bin/open", QStringList() << "-R" << path);
|
|
|
|
}
|
2011-11-09 13:59:25 +01:00
|
|
|
#endif // Q_OS_DARWIN
|
|
|
|
|
2013-06-09 15:52:47 +02:00
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
void ShowFileInExplorer(QString const& path) {
|
2014-02-07 16:34:20 +01:00
|
|
|
QProcess::execute("explorer.exe", QStringList()
|
|
|
|
<< "/select,"
|
|
|
|
<< QDir::toNativeSeparators(path));
|
2013-06-09 15:52:47 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-10-26 14:54:24 +02:00
|
|
|
void OpenInFileBrowser(const QList<QUrl>& urls) {
|
2011-03-17 20:52:21 +01:00
|
|
|
QSet<QString> dirs;
|
|
|
|
|
2014-02-10 14:29:07 +01:00
|
|
|
for (const QUrl& url : urls) {
|
2011-10-26 14:43:28 +02:00
|
|
|
if (url.scheme() != "file") {
|
2011-03-17 20:52:21 +01:00
|
|
|
continue;
|
2011-10-26 14:43:28 +02:00
|
|
|
}
|
|
|
|
QString path = url.toLocalFile();
|
2011-03-17 20:52:21 +01:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
if (!QFile::exists(path)) continue;
|
2011-03-17 20:52:21 +01:00
|
|
|
|
2011-10-26 14:43:28 +02:00
|
|
|
const QString directory = QFileInfo(path).dir().path();
|
2014-02-07 16:34:20 +01:00
|
|
|
if (dirs.contains(directory)) continue;
|
2011-03-17 20:52:21 +01:00
|
|
|
dirs.insert(directory);
|
2013-06-09 16:29:40 +02:00
|
|
|
qLog(Debug) << path;
|
2013-04-20 23:08:45 +02:00
|
|
|
#ifdef Q_OS_DARWIN
|
|
|
|
// revealing multiple files in the finder only opens one window,
|
|
|
|
// so it also makes sense to reveal at most one per directory
|
|
|
|
RevealFileInFinder(path);
|
2013-06-09 15:52:47 +02:00
|
|
|
#elif defined(Q_OS_WIN32)
|
|
|
|
ShowFileInExplorer(path);
|
2013-04-20 23:08:45 +02:00
|
|
|
#else
|
2015-10-11 22:56:42 +02:00
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(directory));
|
2013-04-20 23:08:45 +02:00
|
|
|
#endif
|
2011-03-17 20:52:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
QByteArray Hmac(const QByteArray& key, const QByteArray& data,
|
|
|
|
HashFunction method) {
|
|
|
|
const int kBlockSize = 64; // bytes
|
2011-07-23 20:33:00 +02:00
|
|
|
Q_ASSERT(key.length() <= kBlockSize);
|
|
|
|
|
2014-11-01 19:26:05 +01:00
|
|
|
QByteArray inner_padding(kBlockSize, static_cast<char>(0x36));
|
|
|
|
QByteArray outer_padding(kBlockSize, static_cast<char>(0x5c));
|
2011-07-23 20:33:00 +02:00
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
for (int i = 0; i < key.length(); ++i) {
|
2011-07-23 20:33:00 +02:00
|
|
|
inner_padding[i] = inner_padding[i] ^ key[i];
|
|
|
|
outer_padding[i] = outer_padding[i] ^ key[i];
|
|
|
|
}
|
2011-09-01 23:45:47 +02:00
|
|
|
if (Md5_Algo == method) {
|
2014-02-07 16:34:20 +01:00
|
|
|
return QCryptographicHash::hash(
|
|
|
|
outer_padding + QCryptographicHash::hash(inner_padding + data,
|
|
|
|
QCryptographicHash::Md5),
|
|
|
|
QCryptographicHash::Md5);
|
2012-11-27 18:35:06 +01:00
|
|
|
} else if (Sha1_Algo == method) {
|
2014-02-07 16:34:20 +01:00
|
|
|
return QCryptographicHash::hash(
|
|
|
|
outer_padding + QCryptographicHash::hash(inner_padding + data,
|
|
|
|
QCryptographicHash::Sha1),
|
|
|
|
QCryptographicHash::Sha1);
|
|
|
|
} else { // Sha256_Algo, currently default
|
2018-12-03 01:45:20 +01:00
|
|
|
return QCryptographicHash::hash(
|
|
|
|
outer_padding + QCryptographicHash::hash(inner_padding + data,
|
|
|
|
QCryptographicHash::Sha256),
|
|
|
|
QCryptographicHash::Sha256);
|
2011-09-01 23:45:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray HmacSha256(const QByteArray& key, const QByteArray& data) {
|
|
|
|
return Hmac(key, data, Sha256_Algo);
|
|
|
|
}
|
2011-07-23 20:33:00 +02:00
|
|
|
|
2011-09-01 23:45:47 +02:00
|
|
|
QByteArray HmacMd5(const QByteArray& key, const QByteArray& data) {
|
|
|
|
return Hmac(key, data, Md5_Algo);
|
2011-07-23 20:33:00 +02:00
|
|
|
}
|
|
|
|
|
2012-11-27 18:35:06 +01:00
|
|
|
QByteArray HmacSha1(const QByteArray& key, const QByteArray& data) {
|
|
|
|
return Hmac(key, data, Sha1_Algo);
|
|
|
|
}
|
|
|
|
|
2013-12-31 15:21:50 +01:00
|
|
|
// File must not be open and will be closed afterwards!
|
2014-02-07 16:34:20 +01:00
|
|
|
QByteArray Sha1File(QFile& file) {
|
2013-12-31 15:21:50 +01:00
|
|
|
file.open(QIODevice::ReadOnly);
|
2014-01-18 18:33:49 +01:00
|
|
|
QCryptographicHash hash(QCryptographicHash::Sha1);
|
2013-12-31 15:21:50 +01:00
|
|
|
QByteArray data;
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
while (!file.atEnd()) {
|
|
|
|
data = file.read(1000000); // 1 mib
|
2013-12-31 15:21:50 +01:00
|
|
|
hash.addData(data.data(), data.length());
|
|
|
|
data.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
return hash.result();
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray Sha1CoverHash(const QString& artist, const QString& album) {
|
|
|
|
QCryptographicHash hash(QCryptographicHash::Sha1);
|
|
|
|
hash.addData(artist.toLower().toUtf8().constData());
|
|
|
|
hash.addData(album.toLower().toUtf8().constData());
|
|
|
|
|
|
|
|
return hash.result();
|
|
|
|
}
|
|
|
|
|
2011-07-26 14:02:59 +02:00
|
|
|
QString PrettySize(const QSize& size) {
|
2014-02-07 16:34:20 +01:00
|
|
|
return QString::number(size.width()) + "x" + QString::number(size.height());
|
2011-07-26 14:02:59 +02:00
|
|
|
}
|
|
|
|
|
2011-10-15 21:48:48 +02:00
|
|
|
void ForwardMouseEvent(const QMouseEvent* e, QWidget* target) {
|
|
|
|
QMouseEvent c(e->type(), target->mapFromGlobal(e->globalPos()),
|
|
|
|
e->globalPos(), e->button(), e->buttons(), e->modifiers());
|
|
|
|
|
|
|
|
QApplication::sendEvent(target, &c);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsMouseEventInWidget(const QMouseEvent* e, const QWidget* widget) {
|
|
|
|
return widget->rect().contains(widget->mapFromGlobal(e->globalPos()));
|
|
|
|
}
|
|
|
|
|
2011-11-28 19:11:09 +01:00
|
|
|
quint16 PickUnusedPort() {
|
|
|
|
forever {
|
2021-04-09 07:58:26 +02:00
|
|
|
#if (QT_VERSION < QT_VERSION_CHECK(5, 10, 0))
|
2011-11-28 19:11:09 +01:00
|
|
|
const quint16 port = 49152 + qrand() % 16384;
|
2021-04-09 07:58:26 +02:00
|
|
|
#else
|
|
|
|
const quint16 port = QRandomGenerator::global()->bounded(49152, 65536);
|
|
|
|
#endif
|
2011-11-28 19:11:09 +01:00
|
|
|
|
|
|
|
QTcpServer server;
|
|
|
|
if (server.listen(QHostAddress::Any, port)) {
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-07 22:51:02 +01:00
|
|
|
void ConsumeCurrentElement(QXmlStreamReader* reader) {
|
|
|
|
int level = 1;
|
|
|
|
while (level != 0 && !reader->atEnd()) {
|
|
|
|
switch (reader->readNext()) {
|
2014-02-07 16:34:20 +01:00
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
++level;
|
|
|
|
break;
|
|
|
|
case QXmlStreamReader::EndElement:
|
|
|
|
--level;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2012-01-07 22:51:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-04 21:47:58 +01:00
|
|
|
bool ParseUntilElement(QXmlStreamReader* reader, const QString& name) {
|
|
|
|
while (!reader->atEnd()) {
|
|
|
|
QXmlStreamReader::TokenType type = reader->readNext();
|
|
|
|
switch (type) {
|
|
|
|
case QXmlStreamReader::StartElement:
|
|
|
|
if (reader->name() == name) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDateTime ParseRFC822DateTime(const QString& text) {
|
2017-03-17 17:54:02 +01:00
|
|
|
QRegExp regexp(
|
|
|
|
"(\\d{1,2}) (\\w{3,12}) (\\d+) (\\d{1,2}):(\\d{1,2}):(\\d{1,2})");
|
2015-02-25 15:29:50 +01:00
|
|
|
if (regexp.indexIn(text) == -1) {
|
|
|
|
return QDateTime();
|
2014-12-11 22:42:18 +01:00
|
|
|
}
|
2015-02-25 15:29:50 +01:00
|
|
|
|
2017-03-17 17:54:02 +01:00
|
|
|
enum class MatchNames { DAYS = 1, MONTHS, YEARS, HOURS, MINUTES, SECONDS };
|
2014-12-11 22:42:18 +01:00
|
|
|
|
|
|
|
QMap<QString, int> monthmap;
|
|
|
|
monthmap["Jan"] = 1;
|
|
|
|
monthmap["Feb"] = 2;
|
|
|
|
monthmap["Mar"] = 3;
|
|
|
|
monthmap["Apr"] = 4;
|
|
|
|
monthmap["May"] = 5;
|
|
|
|
monthmap["Jun"] = 6;
|
|
|
|
monthmap["Jul"] = 7;
|
|
|
|
monthmap["Aug"] = 8;
|
|
|
|
monthmap["Sep"] = 9;
|
|
|
|
monthmap["Oct"] = 10;
|
|
|
|
monthmap["Nov"] = 11;
|
|
|
|
monthmap["Dec"] = 12;
|
2015-02-25 15:29:50 +01:00
|
|
|
monthmap["January"] = 1;
|
|
|
|
monthmap["February"] = 2;
|
|
|
|
monthmap["March"] = 3;
|
|
|
|
monthmap["April"] = 4;
|
|
|
|
monthmap["May"] = 5;
|
|
|
|
monthmap["June"] = 6;
|
|
|
|
monthmap["July"] = 7;
|
|
|
|
monthmap["August"] = 8;
|
|
|
|
monthmap["September"] = 9;
|
|
|
|
monthmap["October"] = 10;
|
|
|
|
monthmap["November"] = 11;
|
|
|
|
monthmap["December"] = 12;
|
|
|
|
|
|
|
|
const QDate date(regexp.cap(static_cast<int>(MatchNames::YEARS)).toInt(),
|
|
|
|
monthmap[regexp.cap(static_cast<int>(MatchNames::MONTHS))],
|
|
|
|
regexp.cap(static_cast<int>(MatchNames::DAYS)).toInt());
|
|
|
|
|
|
|
|
const QTime time(regexp.cap(static_cast<int>(MatchNames::HOURS)).toInt(),
|
|
|
|
regexp.cap(static_cast<int>(MatchNames::MINUTES)).toInt(),
|
|
|
|
regexp.cap(static_cast<int>(MatchNames::SECONDS)).toInt());
|
|
|
|
|
2017-03-17 17:54:02 +01:00
|
|
|
return QDateTime(date, time);
|
2012-03-04 21:47:58 +01:00
|
|
|
}
|
|
|
|
|
2012-01-16 15:22:30 +01:00
|
|
|
const char* EnumToString(const QMetaObject& meta, const char* name, int value) {
|
|
|
|
int index = meta.indexOfEnumerator(name);
|
2014-02-07 16:34:20 +01:00
|
|
|
if (index == -1) return "[UnknownEnum]";
|
2012-01-16 15:22:30 +01:00
|
|
|
QMetaEnum metaenum = meta.enumerator(index);
|
|
|
|
const char* result = metaenum.valueToKey(value);
|
2014-02-07 16:34:20 +01:00
|
|
|
if (result == 0) return "[UnknownEnumValue]";
|
2012-01-16 15:22:30 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-03-04 21:47:58 +01:00
|
|
|
QStringList Prepend(const QString& text, const QStringList& list) {
|
|
|
|
QStringList ret(list);
|
2014-02-07 16:34:20 +01:00
|
|
|
for (int i = 0; i < ret.count(); ++i) ret[i].prepend(text);
|
2012-03-04 21:47:58 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList Updateify(const QStringList& list) {
|
|
|
|
QStringList ret(list);
|
2014-02-07 16:34:20 +01:00
|
|
|
for (int i = 0; i < ret.count(); ++i) ret[i].prepend(ret[i] + " = :");
|
2012-03-04 21:47:58 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-03-05 19:15:45 +01:00
|
|
|
QString DecodeHtmlEntities(const QString& text) {
|
|
|
|
QString copy(text);
|
|
|
|
copy.replace("&", "&");
|
|
|
|
copy.replace(""", "\"");
|
|
|
|
copy.replace("'", "'");
|
|
|
|
copy.replace("<", "<");
|
|
|
|
copy.replace(">", ">");
|
|
|
|
return copy;
|
|
|
|
}
|
|
|
|
|
2012-02-26 16:05:46 +01:00
|
|
|
int SetThreadIOPriority(IoPriority priority) {
|
|
|
|
#ifdef Q_OS_LINUX
|
|
|
|
return syscall(SYS_ioprio_set, IOPRIO_WHO_PROCESS, GetThreadId(),
|
|
|
|
4 | priority << IOPRIO_CLASS_SHIFT);
|
|
|
|
#elif defined(Q_OS_DARWIN)
|
|
|
|
return setpriority(PRIO_DARWIN_THREAD, 0,
|
|
|
|
priority == IOPRIO_CLASS_IDLE ? PRIO_DARWIN_BG : 0);
|
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int GetThreadId() {
|
|
|
|
#ifdef Q_OS_LINUX
|
|
|
|
return syscall(SYS_gettid);
|
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-03-09 16:25:49 +01:00
|
|
|
bool IsLaptop() {
|
|
|
|
#ifdef Q_OS_WIN
|
2012-03-09 19:39:40 +01:00
|
|
|
SYSTEM_POWER_STATUS status;
|
2012-03-09 16:25:49 +01:00
|
|
|
if (!GetSystemPowerStatus(&status)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
return !(status.BatteryFlag & 128); // 128 = no system battery
|
2012-10-31 13:25:18 +01:00
|
|
|
#elif defined(Q_OS_LINUX)
|
2014-02-07 16:34:20 +01:00
|
|
|
return !QDir("/proc/acpi/battery")
|
|
|
|
.entryList(QDir::Dirs | QDir::NoDotAndDotDot)
|
|
|
|
.isEmpty();
|
2012-10-31 13:25:18 +01:00
|
|
|
#elif defined(Q_OS_MAC)
|
2012-03-09 17:29:55 +01:00
|
|
|
ScopedCFTypeRef<CFTypeRef> power_sources(IOPSCopyPowerSourcesInfo());
|
|
|
|
ScopedCFTypeRef<CFArrayRef> power_source_list(
|
|
|
|
IOPSCopyPowerSourcesList(power_sources.get()));
|
|
|
|
for (CFIndex i = 0; i < CFArrayGetCount(power_source_list.get()); ++i) {
|
|
|
|
CFTypeRef ps = CFArrayGetValueAtIndex(power_source_list.get(), i);
|
2014-02-07 16:34:20 +01:00
|
|
|
CFDictionaryRef description =
|
|
|
|
IOPSGetPowerSourceDescription(power_sources.get(), ps);
|
2012-03-09 17:29:55 +01:00
|
|
|
|
|
|
|
if (CFDictionaryContainsKey(description, CFSTR(kIOPSBatteryHealthKey))) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2012-03-09 16:25:49 +01:00
|
|
|
return false;
|
2012-10-31 13:25:18 +01:00
|
|
|
#else
|
|
|
|
return false;
|
2012-03-09 16:25:49 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-07-27 07:27:08 +02:00
|
|
|
QString SystemLanguageName() {
|
2014-02-07 16:34:20 +01:00
|
|
|
QString system_language = QLocale::system().uiLanguages().empty()
|
|
|
|
? QLocale::system().name()
|
|
|
|
: QLocale::system().uiLanguages().first();
|
2013-07-27 07:27:08 +02:00
|
|
|
// uiLanguages returns strings with "-" as separators for language/region;
|
|
|
|
// however QTranslator needs "_" separators
|
|
|
|
system_language.replace("-", "_");
|
|
|
|
|
|
|
|
return system_language;
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
bool UrlOnSameDriveAsClementine(const QUrl& url) {
|
|
|
|
if (url.scheme() != "file") return false;
|
2013-10-03 17:08:42 +02:00
|
|
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
QUrl appUrl = QUrl::fromLocalFile(QCoreApplication::applicationDirPath());
|
|
|
|
if (url.toLocalFile().left(1) == appUrl.toLocalFile().left(1))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
// Non windows systems have always a / in the path
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
QUrl GetRelativePathToClementineBin(const QUrl& url) {
|
2020-02-12 07:23:11 +01:00
|
|
|
QString relPath = GetRelativePathToClementineBin(url.toLocalFile());
|
2020-02-12 07:38:52 +01:00
|
|
|
QUrl rel_url = QUrl::fromLocalFile(relPath);
|
|
|
|
// QUrl considers a URL relative if the schema is omitted.
|
|
|
|
rel_url.setScheme(QString());
|
|
|
|
return rel_url;
|
2020-02-12 07:23:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString GetRelativePathToClementineBin(const QString& abspath) {
|
2013-10-03 17:08:42 +02:00
|
|
|
QDir appPath(QCoreApplication::applicationDirPath());
|
2020-02-12 07:23:11 +01:00
|
|
|
return appPath.relativeFilePath(abspath);
|
2013-10-03 17:08:42 +02:00
|
|
|
}
|
|
|
|
|
2014-02-01 03:22:41 +01:00
|
|
|
QString PathWithoutFilenameExtension(const QString& filename) {
|
|
|
|
if (filename.section('/', -1, -1).contains('.'))
|
|
|
|
return filename.section('.', 0, -2);
|
|
|
|
return filename;
|
|
|
|
}
|
|
|
|
|
2014-02-07 16:34:20 +01:00
|
|
|
QString FiddleFileExtension(const QString& filename,
|
|
|
|
const QString& new_extension) {
|
2014-02-01 03:22:41 +01:00
|
|
|
return PathWithoutFilenameExtension(filename) + "." + new_extension;
|
|
|
|
}
|
|
|
|
|
2020-01-29 09:25:39 +01:00
|
|
|
QByteArray GetUriForGstreamer(const QUrl& url) {
|
|
|
|
if (url.scheme() == "file") {
|
|
|
|
QString local_file = url.toLocalFile();
|
|
|
|
if (local_file.indexOf("//") == 0) {
|
|
|
|
// Exclude / from encoding.
|
|
|
|
return QByteArray("file://") + QUrl::toPercentEncoding(local_file, "/");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return url.toEncoded();
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:59:19 +01:00
|
|
|
QString ScrubUrlQueries(const QString& str) {
|
|
|
|
// If the URL isn't followed by whitespace, this will eat extra characters.
|
|
|
|
QRegExp rx("((?:http|https)://\\S*\\?)\\S*");
|
|
|
|
// QString::replace is non const, so operate on a copy.
|
|
|
|
return QString(str).replace(rx, "\\1 (query removed)");
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:32:38 +01:00
|
|
|
QString MakeBugReportUrl(const QString& title) {
|
|
|
|
// Example:
|
|
|
|
// https://github.com/clementine-player/Clementine/issues/new?title=New%20bug
|
|
|
|
QUrl url("https://github.com/clementine-player/Clementine/issues/new");
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("title", title);
|
|
|
|
url.setQuery(query);
|
|
|
|
return url.toString(QUrl::FullyEncoded);
|
|
|
|
}
|
|
|
|
|
2010-12-09 13:34:08 +01:00
|
|
|
} // namespace Utilities
|
2010-08-23 21:13:27 +02:00
|
|
|
|
|
|
|
ScopedWCharArray::ScopedWCharArray(const QString& str)
|
2014-02-07 16:34:20 +01:00
|
|
|
: chars_(str.length()), data_(new wchar_t[chars_ + 1]) {
|
2010-08-23 21:13:27 +02:00
|
|
|
str.toWCharArray(data_.get());
|
|
|
|
data_[chars_] = '\0';
|
|
|
|
}
|