Files
qt-creator/src/libs/utils/classnamevalidatinglineedit.cpp

176 lines
5.0 KiB
C++
Raw Normal View History

// Copyright (C) 2016 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
2008-12-02 14:09:21 +01:00
2008-12-02 12:01:29 +01:00
#include "classnamevalidatinglineedit.h"
#include "qtcassert.h"
2008-12-09 15:25:01 +01:00
#include <QRegularExpression>
2008-12-02 12:01:29 +01:00
/*!
\class Utils::ClassNameValidatingLineEdit
\brief The ClassNameValidatingLineEdit class implements a line edit that
validates a C++ class name and emits a signal
to derive suggested file names from it.
*/
2008-12-02 12:01:29 +01:00
namespace Utils {
// Match something like "Namespace1::Namespace2::ClassName".
2008-12-02 12:01:29 +01:00
struct ClassNameValidatingLineEditPrivate
{
QRegularExpression m_nameRegexp;
QString m_namespaceDelimiter{"::"};
bool m_namespacesEnabled = false;
bool m_lowerCaseFileName = true;
bool m_forceFirstCapitalLetter = false;
2008-12-02 12:01:29 +01:00
};
// --------------------- ClassNameValidatingLineEdit
ClassNameValidatingLineEdit::ClassNameValidatingLineEdit(QWidget *parent) :
FancyLineEdit(parent),
d(new ClassNameValidatingLineEditPrivate)
2008-12-02 12:01:29 +01:00
{
setValidationFunction([this](FancyLineEdit *edit, QString *errorMessage) {
return validateClassName(edit, errorMessage);
});
updateRegExp();
2008-12-02 12:01:29 +01:00
}
ClassNameValidatingLineEdit::~ClassNameValidatingLineEdit()
{
delete d;
2008-12-02 12:01:29 +01:00
}
bool ClassNameValidatingLineEdit::namespacesEnabled() const
{
return d->m_namespacesEnabled;
2008-12-02 12:01:29 +01:00
}
void ClassNameValidatingLineEdit::setNamespacesEnabled(bool b)
{
d->m_namespacesEnabled = b;
2008-12-02 12:01:29 +01:00
}
/**
* @return Language-specific namespace delimiter, e.g. '::' or '.'
*/
QString ClassNameValidatingLineEdit::namespaceDelimiter()
{
return d->m_namespaceDelimiter;
}
/**
* @brief Sets language-specific namespace delimiter, e.g. '::' or '.'
* Do not use identifier characters in delimiter
*/
void ClassNameValidatingLineEdit::setNamespaceDelimiter(const QString &delimiter)
{
d->m_namespaceDelimiter = delimiter;
}
bool ClassNameValidatingLineEdit::validateClassName(FancyLineEdit *edit, QString *errorMessage) const
2008-12-02 12:01:29 +01:00
{
QTC_ASSERT(d->m_nameRegexp.isValid(), return false);
const QString value = edit->text();
if (!d->m_namespacesEnabled && value.contains(d->m_namespaceDelimiter)) {
2008-12-02 12:01:29 +01:00
if (errorMessage)
*errorMessage = tr("The class name must not contain namespace delimiters.");
return false;
} else if (value.isEmpty()) {
if (errorMessage)
*errorMessage = tr("Please enter a class name.");
return false;
} else if (!d->m_nameRegexp.match(value).hasMatch()) {
2008-12-02 12:01:29 +01:00
if (errorMessage)
*errorMessage = tr("The class name contains invalid characters.");
return false;
}
return true;
}
void ClassNameValidatingLineEdit::handleChanged(const QString &t)
2008-12-02 12:01:29 +01:00
{
if (isValid()) {
// Suggest file names, strip namespaces
QString fileName = d->m_lowerCaseFileName ? t.toLower() : t;
if (d->m_namespacesEnabled) {
const int namespaceIndex = fileName.lastIndexOf(d->m_namespaceDelimiter);
2008-12-02 12:01:29 +01:00
if (namespaceIndex != -1)
fileName.remove(0, namespaceIndex + d->m_namespaceDelimiter.size());
2008-12-02 12:01:29 +01:00
}
emit updateFileName(fileName);
}
}
QString ClassNameValidatingLineEdit::fixInputString(const QString &string)
{
if (!forceFirstCapitalLetter())
return string;
QString fixedString = string;
if (!string.isEmpty() && string.at(0).isLower())
fixedString[0] = string.at(0).toUpper();
return fixedString;
}
void ClassNameValidatingLineEdit::updateRegExp() const
{
const QString pattern = "^%1(%2%1)*$";
d->m_nameRegexp.setPattern(pattern.arg("[a-zA-Z_][a-zA-Z0-9_]*")
.arg(QRegularExpression::escape(d->m_namespaceDelimiter)));
}
2008-12-02 12:01:29 +01:00
QString ClassNameValidatingLineEdit::createClassName(const QString &name)
{
// Remove spaces and convert the adjacent characters to uppercase
QString className = name;
const QRegularExpression spaceMatcher(" +(\\w)");
QTC_CHECK(spaceMatcher.isValid());
while (true) {
const QRegularExpressionMatch match = spaceMatcher.match(className);
if (!match.hasMatch())
break;
className.replace(match.capturedStart(), match.capturedLength(), match.captured(1).toUpper());
2008-12-02 12:01:29 +01:00
}
// Filter out any remaining invalid characters
className.remove(QRegularExpression("[^a-zA-Z0-9_]"));
2008-12-02 12:01:29 +01:00
// If the first character is numeric, prefix the name with a "_"
if (className.at(0).isNumber()) {
className.prepend(QLatin1Char('_'));
} else {
// Convert the first character to uppercase
className.replace(0, 1, className.left(1).toUpper());
}
return className;
}
bool ClassNameValidatingLineEdit::lowerCaseFileName() const
{
return d->m_lowerCaseFileName;
}
void ClassNameValidatingLineEdit::setLowerCaseFileName(bool v)
{
d->m_lowerCaseFileName = v;
}
bool ClassNameValidatingLineEdit::forceFirstCapitalLetter() const
{
return d->m_forceFirstCapitalLetter;
}
void ClassNameValidatingLineEdit::setForceFirstCapitalLetter(bool b)
{
d->m_forceFirstCapitalLetter = b;
}
2008-12-02 14:09:21 +01:00
} // namespace Utils