2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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
|
2016-01-15 14:58:39 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** 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.
|
2010-12-17 16:01:08 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2009-11-25 12:34:56 +01:00
|
|
|
#include "stringutils.h"
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2012-08-23 15:53:58 +02:00
|
|
|
#include "hostosinfo.h"
|
|
|
|
|
|
2014-12-05 12:23:00 +01:00
|
|
|
#include <utils/algorithm.h>
|
|
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDir>
|
2009-11-25 12:34:56 +01:00
|
|
|
|
|
|
|
|
#include <limits.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
namespace Utils {
|
|
|
|
|
|
2009-05-08 12:09:21 +02:00
|
|
|
QTCREATOR_UTILS_EXPORT QString settingsKey(const QString &category)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
QString rc(category);
|
|
|
|
|
const QChar underscore = QLatin1Char('_');
|
2010-01-13 14:21:39 +01:00
|
|
|
// Remove the sort category "X.Category" -> "Category"
|
|
|
|
|
if (rc.size() > 2 && rc.at(0).isLetter() && rc.at(1) == QLatin1Char('.'))
|
|
|
|
|
rc.remove(0, 2);
|
|
|
|
|
// Replace special characters
|
2008-12-02 12:01:29 +01:00
|
|
|
const int size = rc.size();
|
2008-12-02 14:09:21 +01:00
|
|
|
for (int i = 0; i < size; i++) {
|
2008-12-02 12:01:29 +01:00
|
|
|
const QChar c = rc.at(i);
|
2008-12-02 14:09:21 +01:00
|
|
|
if (!c.isLetterOrNumber() && c != underscore)
|
2008-12-02 12:01:29 +01:00
|
|
|
rc[i] = underscore;
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2009-11-25 12:34:56 +01:00
|
|
|
// Figure out length of common start of string ("C:\a", "c:\b" -> "c:\"
|
|
|
|
|
static inline int commonPartSize(const QString &s1, const QString &s2)
|
|
|
|
|
{
|
|
|
|
|
const int size = qMin(s1.size(), s2.size());
|
|
|
|
|
for (int i = 0; i < size; i++)
|
|
|
|
|
if (s1.at(i) != s2.at(i))
|
|
|
|
|
return i;
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QString commonPrefix(const QStringList &strings)
|
|
|
|
|
{
|
|
|
|
|
switch (strings.size()) {
|
|
|
|
|
case 0:
|
|
|
|
|
return QString();
|
|
|
|
|
case 1:
|
|
|
|
|
return strings.front();
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
// Figure out common string part: "C:\foo\bar1" "C:\foo\bar2" -> "C:\foo\bar"
|
|
|
|
|
int commonLength = INT_MAX;
|
|
|
|
|
const int last = strings.size() - 1;
|
|
|
|
|
for (int i = 0; i < last; i++)
|
|
|
|
|
commonLength = qMin(commonLength, commonPartSize(strings.at(i), strings.at(i + 1)));
|
|
|
|
|
if (!commonLength)
|
|
|
|
|
return QString();
|
|
|
|
|
return strings.at(0).left(commonLength);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QString commonPath(const QStringList &files)
|
|
|
|
|
{
|
2014-12-05 12:23:00 +01:00
|
|
|
QStringList appendedSlashes = Utils::transform(files, [](const QString &file) -> QString {
|
|
|
|
|
if (!file.endsWith(QLatin1Char('/')))
|
|
|
|
|
return QString(file + QLatin1Char('/'));
|
|
|
|
|
return file;
|
|
|
|
|
});
|
|
|
|
|
QString common = commonPrefix(appendedSlashes);
|
2009-11-25 12:34:56 +01:00
|
|
|
// Find common directory part: "C:\foo\bar" -> "C:\foo"
|
|
|
|
|
int lastSeparatorPos = common.lastIndexOf(QLatin1Char('/'));
|
|
|
|
|
if (lastSeparatorPos == -1)
|
|
|
|
|
lastSeparatorPos = common.lastIndexOf(QLatin1Char('\\'));
|
|
|
|
|
if (lastSeparatorPos == -1)
|
|
|
|
|
return QString();
|
2012-08-23 15:53:58 +02:00
|
|
|
if (HostOsInfo::isAnyUnixHost() && lastSeparatorPos == 0) // Unix: "/a", "/b" -> '/'
|
2009-11-25 12:34:56 +01:00
|
|
|
lastSeparatorPos = 1;
|
|
|
|
|
common.truncate(lastSeparatorPos);
|
|
|
|
|
return common;
|
|
|
|
|
}
|
|
|
|
|
|
2010-07-13 16:42:36 +02:00
|
|
|
QTCREATOR_UTILS_EXPORT QString withTildeHomePath(const QString &path)
|
|
|
|
|
{
|
2012-08-23 15:53:58 +02:00
|
|
|
if (HostOsInfo::isWindowsHost())
|
|
|
|
|
return path;
|
|
|
|
|
|
2010-07-13 16:42:36 +02:00
|
|
|
static const QString homePath = QDir::homePath();
|
|
|
|
|
|
|
|
|
|
QFileInfo fi(QDir::cleanPath(path));
|
|
|
|
|
QString outPath = fi.absoluteFilePath();
|
|
|
|
|
if (outPath.startsWith(homePath))
|
|
|
|
|
outPath = QLatin1Char('~') + outPath.mid(homePath.size());
|
|
|
|
|
else
|
|
|
|
|
outPath = path;
|
|
|
|
|
return outPath;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-12 19:14:54 +02:00
|
|
|
bool AbstractMacroExpander::expandNestedMacros(const QString &str, int *pos, QString *ret)
|
|
|
|
|
{
|
|
|
|
|
QString varName;
|
|
|
|
|
QChar prev;
|
|
|
|
|
QChar c;
|
|
|
|
|
|
|
|
|
|
int i = *pos;
|
|
|
|
|
int strLen = str.length();
|
|
|
|
|
varName.reserve(strLen - i);
|
|
|
|
|
for (; i < strLen; prev = c) {
|
|
|
|
|
c = str.at(i++);
|
|
|
|
|
if (c == QLatin1Char('}')) {
|
|
|
|
|
if (varName.isEmpty()) { // replace "%{}" with "%"
|
|
|
|
|
*ret = QString(QLatin1Char('%'));
|
|
|
|
|
*pos = i;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (resolveMacro(varName, ret)) {
|
|
|
|
|
*pos = i;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
} else if (c == QLatin1Char('{') && prev == QLatin1Char('%')) {
|
|
|
|
|
if (!expandNestedMacros(str, &i, ret))
|
|
|
|
|
return false;
|
|
|
|
|
varName.chop(1);
|
|
|
|
|
varName += ret;
|
|
|
|
|
} else {
|
|
|
|
|
varName += c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-24 16:17:58 +02:00
|
|
|
int AbstractMacroExpander::findMacro(const QString &str, int *pos, QString *ret)
|
2010-11-08 21:09:19 +01:00
|
|
|
{
|
|
|
|
|
forever {
|
|
|
|
|
int openPos = str.indexOf(QLatin1String("%{"), *pos);
|
|
|
|
|
if (openPos < 0)
|
|
|
|
|
return 0;
|
|
|
|
|
int varPos = openPos + 2;
|
2014-09-12 19:14:54 +02:00
|
|
|
if (expandNestedMacros(str, &varPos, ret)) {
|
2010-11-08 21:09:19 +01:00
|
|
|
*pos = openPos;
|
2014-09-12 19:14:54 +02:00
|
|
|
return varPos - openPos;
|
2010-11-08 21:09:19 +01:00
|
|
|
}
|
|
|
|
|
// An actual expansion may be nested into a "false" one,
|
|
|
|
|
// so we continue right after the last %{.
|
2014-09-12 19:14:54 +02:00
|
|
|
*pos = openPos + 2;
|
2010-11-08 21:09:19 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void expandMacros(QString *str, AbstractMacroExpander *mx)
|
|
|
|
|
{
|
|
|
|
|
QString rsts;
|
|
|
|
|
|
|
|
|
|
for (int pos = 0; int len = mx->findMacro(*str, &pos, &rsts); ) {
|
|
|
|
|
str->replace(pos, len, rsts);
|
|
|
|
|
pos += rsts.length();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QString expandMacros(const QString &str, AbstractMacroExpander *mx)
|
|
|
|
|
{
|
|
|
|
|
QString ret = str;
|
|
|
|
|
expandMacros(&ret, mx);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 14:09:21 +01:00
|
|
|
} // namespace Utils
|