2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2011-07-08 09:56:02 +02:00
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
|
|
|
** Contact: http://www.qt.io/licensing
|
2011-07-08 09:56:02 +02:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2011-07-08 09:56:02 +02: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
|
2015-01-14 18:07:15 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms and
|
|
|
|
|
** conditions see http://www.qt.io/terms-conditions. For further information
|
2014-10-01 13:21:18 +02:00
|
|
|
** use the contact form at http://www.qt.io/contact-us.
|
2011-07-08 09:56:02 +02:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2012-10-02 09:12:39 +02:00
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
2011-07-08 09:56:02 +02:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2011-07-08 09:56:02 +02:00
|
|
|
|
|
|
|
|
#include "cpptoolsreuse.h"
|
|
|
|
|
|
2015-06-02 17:25:41 +02:00
|
|
|
#include "cpptoolsplugin.h"
|
|
|
|
|
|
2015-07-10 14:57:42 +02:00
|
|
|
#include <coreplugin/documentmanager.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
2015-02-26 13:38:54 +01:00
|
|
|
#include <coreplugin/idocument.h>
|
2014-06-10 16:05:14 -04:00
|
|
|
#include <texteditor/convenience.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
|
2011-11-14 13:14:39 +01:00
|
|
|
#include <cplusplus/Overview.h>
|
|
|
|
|
#include <cplusplus/LookupContext.h>
|
2015-11-03 13:52:52 +01:00
|
|
|
#include <utils/algorithm.h>
|
2014-05-19 11:17:39 -04:00
|
|
|
#include <utils/qtcassert.h>
|
2011-11-14 13:14:39 +01:00
|
|
|
|
2015-05-12 14:20:32 +02:00
|
|
|
#include <QDebug>
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QSet>
|
2010-11-03 11:02:25 +01:00
|
|
|
#include <QStringRef>
|
2014-06-10 16:05:14 -04:00
|
|
|
#include <QTextCursor>
|
|
|
|
|
#include <QTextDocument>
|
2014-05-19 11:17:39 -04:00
|
|
|
|
2011-11-14 13:14:39 +01:00
|
|
|
using namespace CPlusPlus;
|
|
|
|
|
|
2011-07-08 09:56:02 +02:00
|
|
|
namespace CppTools {
|
|
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
static void moveCursorToStartOrEndOfIdentifier(QTextCursor *tc,
|
|
|
|
|
QTextCursor::MoveOperation op,
|
|
|
|
|
int posDiff = 0)
|
|
|
|
|
{
|
2011-07-08 09:56:02 +02:00
|
|
|
QTextDocument *doc = tc->document();
|
|
|
|
|
if (!doc)
|
|
|
|
|
return;
|
|
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
QChar ch = doc->characterAt(tc->position() - posDiff);
|
2014-05-08 09:48:27 -04:00
|
|
|
while (isValidIdentifierChar(ch)) {
|
2012-02-16 15:09:56 +01:00
|
|
|
tc->movePosition(op);
|
|
|
|
|
ch = doc->characterAt(tc->position() - posDiff);
|
2011-07-08 09:56:02 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-16 15:09:56 +01:00
|
|
|
void moveCursorToEndOfIdentifier(QTextCursor *tc)
|
|
|
|
|
{
|
|
|
|
|
moveCursorToStartOrEndOfIdentifier(tc, QTextCursor::NextCharacter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void moveCursorToStartOfIdentifier(QTextCursor *tc)
|
|
|
|
|
{
|
|
|
|
|
moveCursorToStartOrEndOfIdentifier(tc, QTextCursor::PreviousCharacter, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-14 13:14:39 +01:00
|
|
|
static bool isOwnershipRAIIName(const QString &name)
|
|
|
|
|
{
|
|
|
|
|
static QSet<QString> knownNames;
|
|
|
|
|
if (knownNames.isEmpty()) {
|
|
|
|
|
// Qt
|
|
|
|
|
knownNames.insert(QLatin1String("QScopedPointer"));
|
|
|
|
|
knownNames.insert(QLatin1String("QScopedArrayPointer"));
|
|
|
|
|
knownNames.insert(QLatin1String("QMutexLocker"));
|
|
|
|
|
knownNames.insert(QLatin1String("QReadLocker"));
|
|
|
|
|
knownNames.insert(QLatin1String("QWriteLocker"));
|
|
|
|
|
// Standard C++
|
|
|
|
|
knownNames.insert(QLatin1String("auto_ptr"));
|
|
|
|
|
knownNames.insert(QLatin1String("unique_ptr"));
|
|
|
|
|
// Boost
|
|
|
|
|
knownNames.insert(QLatin1String("scoped_ptr"));
|
|
|
|
|
knownNames.insert(QLatin1String("scoped_array"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return knownNames.contains(name);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-04 17:01:07 +02:00
|
|
|
bool isOwnershipRAIIType(Symbol *symbol, const LookupContext &context)
|
2011-11-14 13:14:39 +01:00
|
|
|
{
|
|
|
|
|
if (!symbol)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
// This is not a "real" comparison of types. What we do is to resolve the symbol
|
|
|
|
|
// in question and then try to match its name with already known ones.
|
|
|
|
|
if (symbol->isDeclaration()) {
|
|
|
|
|
Declaration *declaration = symbol->asDeclaration();
|
|
|
|
|
const NamedType *namedType = declaration->type()->asNamedType();
|
|
|
|
|
if (namedType) {
|
2015-11-19 13:49:26 +01:00
|
|
|
ClassOrNamespace *clazz = context.lookupType(namedType->name(),
|
2011-11-14 13:14:39 +01:00
|
|
|
declaration->enclosingScope());
|
|
|
|
|
if (clazz && !clazz->symbols().isEmpty()) {
|
|
|
|
|
Overview overview;
|
|
|
|
|
Symbol *symbol = clazz->symbols().at(0);
|
|
|
|
|
return isOwnershipRAIIName(overview.prettyName(symbol->name()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-08 09:48:27 -04:00
|
|
|
bool isValidAsciiIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
2014-09-22 10:30:39 +02:00
|
|
|
return ch.isLetterOrNumber() || ch == QLatin1Char('_');
|
2014-05-08 09:48:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isValidFirstIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
|
|
|
|
return ch.isLetter() || ch == QLatin1Char('_') || ch.isHighSurrogate() || ch.isLowSurrogate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool isValidIdentifierChar(const QChar &ch)
|
|
|
|
|
{
|
|
|
|
|
return isValidFirstIdentifierChar(ch) || ch.isNumber();
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-21 18:24:14 +01:00
|
|
|
bool isValidIdentifier(const QString &s)
|
|
|
|
|
{
|
|
|
|
|
const int length = s.length();
|
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
|
const QChar &c = s.at(i);
|
|
|
|
|
if (i == 0) {
|
2014-05-08 09:48:27 -04:00
|
|
|
if (!isValidFirstIdentifierChar(c))
|
2011-12-21 18:24:14 +01:00
|
|
|
return false;
|
|
|
|
|
} else {
|
2014-05-08 09:48:27 -04:00
|
|
|
if (!isValidIdentifierChar(c))
|
2011-12-21 18:24:14 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-03 11:02:25 +01:00
|
|
|
bool isQtKeyword(const QStringRef &text)
|
|
|
|
|
{
|
|
|
|
|
switch (text.length()) {
|
|
|
|
|
case 4:
|
|
|
|
|
switch (text.at(0).toLatin1()) {
|
|
|
|
|
case 'e':
|
|
|
|
|
if (text == QLatin1String("emit"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
case 'S':
|
|
|
|
|
if (text == QLatin1String("SLOT"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
if (text.at(0) == QLatin1Char('s') && text == QLatin1String("slots"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 6:
|
|
|
|
|
if (text.at(0) == QLatin1Char('S') && text == QLatin1String("SIGNAL"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
|
|
|
|
switch (text.at(0).toLatin1()) {
|
|
|
|
|
case 's':
|
|
|
|
|
if (text == QLatin1String("signals"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
case 'f':
|
|
|
|
|
if (text == QLatin1String("foreach") || text == QLatin1String("forever"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2011-11-14 13:14:39 +01:00
|
|
|
|
2014-05-19 11:17:39 -04:00
|
|
|
void switchHeaderSource()
|
|
|
|
|
{
|
|
|
|
|
const Core::IDocument *currentDocument = Core::EditorManager::currentDocument();
|
|
|
|
|
QTC_ASSERT(currentDocument, return);
|
2014-12-21 21:54:30 +02:00
|
|
|
const QString otherFile = correspondingHeaderOrSource(currentDocument->filePath().toString());
|
2014-05-19 11:17:39 -04:00
|
|
|
if (!otherFile.isEmpty())
|
|
|
|
|
Core::EditorManager::openEditor(otherFile);
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-10 16:05:14 -04:00
|
|
|
QString identifierUnderCursor(QTextCursor *cursor)
|
|
|
|
|
{
|
|
|
|
|
cursor->movePosition(QTextCursor::StartOfWord);
|
|
|
|
|
cursor->movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor);
|
|
|
|
|
return cursor->selectedText();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const Macro *findCanonicalMacro(const QTextCursor &cursor, Document::Ptr document)
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(document, return 0);
|
|
|
|
|
|
|
|
|
|
int line, column;
|
|
|
|
|
TextEditor::Convenience::convertPosition(cursor.document(), cursor.position(), &line, &column);
|
|
|
|
|
|
|
|
|
|
if (const Macro *macro = document->findMacroDefinitionAt(line)) {
|
|
|
|
|
QTextCursor macroCursor = cursor;
|
|
|
|
|
const QByteArray name = CppTools::identifierUnderCursor(¯oCursor).toUtf8();
|
|
|
|
|
if (macro->name() == name)
|
|
|
|
|
return macro;
|
|
|
|
|
} else if (const Document::MacroUse *use = document->findMacroUseAt(cursor.position())) {
|
|
|
|
|
return &use->macro();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-23 13:44:50 +01:00
|
|
|
TextEditor::TextEditorWidget::Link linkToSymbol(Symbol *symbol)
|
|
|
|
|
{
|
|
|
|
|
typedef TextEditor::TextEditorWidget::Link Link;
|
|
|
|
|
|
|
|
|
|
if (!symbol)
|
|
|
|
|
return Link();
|
|
|
|
|
|
|
|
|
|
const QString filename = QString::fromUtf8(symbol->fileName(),
|
|
|
|
|
symbol->fileNameLength());
|
|
|
|
|
|
|
|
|
|
unsigned line = symbol->line();
|
|
|
|
|
unsigned column = symbol->column();
|
|
|
|
|
|
|
|
|
|
if (column)
|
|
|
|
|
--column;
|
|
|
|
|
|
|
|
|
|
if (symbol->isGenerated())
|
|
|
|
|
column = 0;
|
|
|
|
|
|
|
|
|
|
return Link(filename, line, column);
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-02 17:25:41 +02:00
|
|
|
QSharedPointer<CppCodeModelSettings> codeModelSettings()
|
|
|
|
|
{
|
|
|
|
|
return CppTools::Internal::CppToolsPlugin::instance()->codeModelSettings();
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-12 14:20:32 +02:00
|
|
|
int fileSizeLimit()
|
|
|
|
|
{
|
|
|
|
|
static const QByteArray fileSizeLimitAsByteArray = qgetenv("QTC_CPP_FILE_SIZE_LIMIT_MB");
|
|
|
|
|
static int fileSizeLimitAsInt = -1;
|
|
|
|
|
|
|
|
|
|
if (fileSizeLimitAsInt == -1) {
|
|
|
|
|
bool ok;
|
|
|
|
|
const int limit = fileSizeLimitAsByteArray.toInt(&ok);
|
|
|
|
|
fileSizeLimitAsInt = ok && limit >= 0 ? limit : 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return fileSizeLimitAsInt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool skipFileDueToSizeLimit(const QFileInfo &fileInfo, int limitInMB)
|
|
|
|
|
{
|
|
|
|
|
if (limitInMB == 0) // unlimited
|
|
|
|
|
return false;
|
|
|
|
|
|
2015-09-24 10:35:15 +02:00
|
|
|
const int fileSizeInMB = fileInfo.size() / (1000 * 1000);
|
2015-05-12 14:20:32 +02:00
|
|
|
if (fileSizeInMB > limitInMB) {
|
|
|
|
|
qWarning() << "Files to process limited by QTC_CPP_FILE_SIZE_LIMIT_MB, skipping"
|
|
|
|
|
<< fileInfo.absoluteFilePath();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-08 09:56:02 +02:00
|
|
|
} // CppTools
|