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
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "filenamevalidatinglineedit.h"
|
2008-12-09 15:25:01 +01:00
|
|
|
#include "qtcassert.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QRegExp>
|
|
|
|
|
#include <QDebug>
|
2008-12-05 10:39:08 +01:00
|
|
|
|
2011-03-02 17:13:33 +01:00
|
|
|
/*!
|
|
|
|
|
\class Utils::FileNameValidatingLineEdit
|
|
|
|
|
|
2013-06-05 14:29:24 +02:00
|
|
|
\brief The FileNameValidatingLineEdit class is a control that lets the user
|
|
|
|
|
choose a (base) file name, based on a QLineEdit.
|
|
|
|
|
|
|
|
|
|
The class has
|
2011-03-02 17:13:33 +01:00
|
|
|
some validation logic for embedding into QWizardPage.
|
|
|
|
|
*/
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
namespace Utils {
|
|
|
|
|
|
2009-04-17 20:56:48 +02:00
|
|
|
#define WINDOWS_DEVICES "CON|AUX|PRN|COM1|COM2|LPT1|LPT2|NUL"
|
|
|
|
|
|
2008-12-05 10:39:08 +01:00
|
|
|
// Naming a file like a device name will break on Windows, even if it is
|
|
|
|
|
// "com1.txt". Since we are cross-platform, we generally disallow such file
|
|
|
|
|
// names.
|
2017-11-11 12:42:01 +01:00
|
|
|
static const QRegExp &windowsDeviceNoSubDirPattern()
|
2008-12-05 10:39:08 +01:00
|
|
|
{
|
2017-11-11 12:42:01 +01:00
|
|
|
static const QRegExp rc(QLatin1String(WINDOWS_DEVICES), Qt::CaseInsensitive);
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(rc.isValid(), return rc);
|
2008-12-05 10:39:08 +01:00
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-11 12:42:01 +01:00
|
|
|
static const QRegExp &windowsDeviceSubDirPattern()
|
2008-12-05 10:39:08 +01:00
|
|
|
{
|
2017-11-11 12:42:01 +01:00
|
|
|
static const QRegExp rc(QLatin1String(".*[/\\\\](" WINDOWS_DEVICES ")"), Qt::CaseInsensitive);
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(rc.isValid(), return rc);
|
2008-12-05 10:39:08 +01:00
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ----------- FileNameValidatingLineEdit
|
2008-12-02 12:01:29 +01:00
|
|
|
FileNameValidatingLineEdit::FileNameValidatingLineEdit(QWidget *parent) :
|
2014-02-13 16:13:54 +01:00
|
|
|
FancyLineEdit(parent),
|
2012-03-19 16:12:51 +01:00
|
|
|
m_allowDirectories(false),
|
|
|
|
|
m_forceFirstCapitalLetter(false)
|
2008-12-05 10:39:08 +01:00
|
|
|
{
|
2015-04-28 14:49:56 +02:00
|
|
|
setValidationFunction([this](FancyLineEdit *edit, QString *errorMessage) {
|
|
|
|
|
return validateFileNameExtension(edit->text(), requiredExtensions(), errorMessage)
|
|
|
|
|
&& validateFileName(edit->text(), allowDirectories(), errorMessage);
|
|
|
|
|
});
|
2008-12-05 10:39:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool FileNameValidatingLineEdit::allowDirectories() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2008-12-05 10:39:08 +01:00
|
|
|
return m_allowDirectories;
|
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-05 10:39:08 +01:00
|
|
|
void FileNameValidatingLineEdit::setAllowDirectories(bool v)
|
|
|
|
|
{
|
|
|
|
|
m_allowDirectories = v;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2012-03-19 16:12:51 +01:00
|
|
|
bool FileNameValidatingLineEdit::forceFirstCapitalLetter() const
|
|
|
|
|
{
|
|
|
|
|
return m_forceFirstCapitalLetter;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FileNameValidatingLineEdit::setForceFirstCapitalLetter(bool b)
|
|
|
|
|
{
|
|
|
|
|
m_forceFirstCapitalLetter = b;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
/* Validate a file base name, check for forbidden characters/strings. */
|
|
|
|
|
|
2012-03-27 11:46:32 +02:00
|
|
|
|
|
|
|
|
#define SLASHES "/\\"
|
|
|
|
|
|
2019-02-04 13:45:52 +01:00
|
|
|
static const char notAllowedCharsSubDir[] = ",^@={}[]~!?:&*\"|#%<>$\"'();`' ";
|
2014-01-27 11:13:13 +01:00
|
|
|
static const char notAllowedCharsNoSubDir[] = ",^@={}[]~!?:&*\"|#%<>$\"'();`' " SLASHES;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2008-12-05 10:39:08 +01:00
|
|
|
static const char *notAllowedSubStrings[] = {".."};
|
|
|
|
|
|
|
|
|
|
bool FileNameValidatingLineEdit::validateFileName(const QString &name,
|
|
|
|
|
bool allowDirectories,
|
|
|
|
|
QString *errorMessage /* = 0*/)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
if (name.isEmpty()) {
|
|
|
|
|
if (errorMessage)
|
2010-03-10 10:51:15 +01:00
|
|
|
*errorMessage = tr("Name is empty.");
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Characters
|
2008-12-05 10:39:08 +01:00
|
|
|
const char *notAllowedChars = allowDirectories ? notAllowedCharsSubDir : notAllowedCharsNoSubDir;
|
2008-12-02 12:01:29 +01:00
|
|
|
for (const char *c = notAllowedChars; *c; c++)
|
|
|
|
|
if (name.contains(QLatin1Char(*c))) {
|
2010-03-10 10:51:15 +01:00
|
|
|
if (errorMessage) {
|
2012-01-04 17:34:08 +01:00
|
|
|
const QChar qc = QLatin1Char(*c);
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
if (qc.isSpace())
|
2010-03-10 10:51:15 +01:00
|
|
|
*errorMessage = tr("Name contains white space.");
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
else
|
2014-04-17 14:09:47 +02:00
|
|
|
*errorMessage = tr("Invalid character \"%1\".").arg(qc);
|
2010-03-10 10:51:15 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Substrings
|
|
|
|
|
const int notAllowedSubStringCount = sizeof(notAllowedSubStrings)/sizeof(const char *);
|
|
|
|
|
for (int s = 0; s < notAllowedSubStringCount; s++) {
|
|
|
|
|
const QLatin1String notAllowedSubString(notAllowedSubStrings[s]);
|
|
|
|
|
if (name.contains(notAllowedSubString)) {
|
|
|
|
|
if (errorMessage)
|
2014-04-17 14:09:47 +02:00
|
|
|
*errorMessage = tr("Invalid characters \"%1\".").arg(QString(notAllowedSubString));
|
2008-12-02 12:01:29 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-12-05 10:39:08 +01:00
|
|
|
// Windows devices
|
|
|
|
|
bool matchesWinDevice = windowsDeviceNoSubDirPattern().exactMatch(name);
|
|
|
|
|
if (!matchesWinDevice && allowDirectories)
|
|
|
|
|
matchesWinDevice = windowsDeviceSubDirPattern().exactMatch(name);
|
|
|
|
|
if (matchesWinDevice) {
|
|
|
|
|
if (errorMessage)
|
2010-03-10 10:51:15 +01:00
|
|
|
*errorMessage = tr("Name matches MS Windows device. (%1).").
|
2008-12-05 10:39:08 +01:00
|
|
|
arg(windowsDeviceNoSubDirPattern().pattern().replace(QLatin1Char('|'), QLatin1Char(',')));
|
|
|
|
|
return false;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2012-03-19 16:12:51 +01:00
|
|
|
QString FileNameValidatingLineEdit::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;
|
|
|
|
|
}
|
2011-09-20 12:54:18 +02:00
|
|
|
|
|
|
|
|
bool FileNameValidatingLineEdit::validateFileNameExtension(const QString &fileName,
|
|
|
|
|
const QStringList &requiredExtensions,
|
|
|
|
|
QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
// file extension
|
|
|
|
|
if (!requiredExtensions.isEmpty()) {
|
|
|
|
|
foreach (const QString &requiredExtension, requiredExtensions) {
|
2014-08-29 14:00:18 +02:00
|
|
|
QString extension = QLatin1Char('.') + requiredExtension;
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
if (fileName.endsWith(extension, Qt::CaseSensitive) && extension.count() < fileName.count())
|
2011-09-20 12:54:18 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (errorMessage) {
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
if (requiredExtensions.count() == 1)
|
2011-09-20 12:54:18 +02:00
|
|
|
*errorMessage = tr("File extension %1 is required:").arg(requiredExtensions.first());
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
else
|
2012-01-04 17:34:08 +01:00
|
|
|
*errorMessage = tr("File extensions %1 are required:").arg(requiredExtensions.join(QLatin1String(", ")));
|
2011-09-20 12:54:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList FileNameValidatingLineEdit::requiredExtensions() const
|
|
|
|
|
{
|
|
|
|
|
return m_requiredExtensionList;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FileNameValidatingLineEdit::setRequiredExtensions(const QStringList &extensions)
|
|
|
|
|
{
|
|
|
|
|
m_requiredExtensionList = extensions;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-02 14:09:21 +01:00
|
|
|
} // namespace Utils
|