forked from qt-creator/qt-creator
		
	Change-Id: Ib5423fdd064e4546f848c0b640b0ed0514c26d3a Reviewed-by: Leena Miettinen <riitta-leena.miettinen@digia.com> Reviewed-by: Kai Koehne <kai.koehne@digia.com>
		
			
				
	
	
		
			636 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			636 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/****************************************************************************
 | 
						|
**
 | 
						|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
 | 
						|
** Contact: http://www.qt-project.org/legal
 | 
						|
**
 | 
						|
** This file is part of Qt Creator.
 | 
						|
**
 | 
						|
** 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
 | 
						|
** a written agreement between you and Digia.  For licensing terms and
 | 
						|
** conditions see http://qt.digia.com/licensing.  For further information
 | 
						|
** use the contact form at http://qt.digia.com/contact-us.
 | 
						|
**
 | 
						|
** GNU Lesser General Public License Usage
 | 
						|
** Alternatively, this file may be used under the terms of the GNU Lesser
 | 
						|
** General Public License version 2.1 as published by the Free Software
 | 
						|
** Foundation and appearing in the file LICENSE.LGPL included in the
 | 
						|
** packaging of this file.  Please review the following information to
 | 
						|
** ensure the GNU Lesser General Public License version 2.1 requirements
 | 
						|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | 
						|
**
 | 
						|
** In addition, as a special exception, Digia gives you certain additional
 | 
						|
** rights.  These rights are described in the Digia Qt LGPL Exception
 | 
						|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
 | 
						|
**
 | 
						|
****************************************************************************/
 | 
						|
 | 
						|
#include "insertionpointlocator.h"
 | 
						|
 | 
						|
#include "cpptoolsreuse.h"
 | 
						|
#include "symbolfinder.h"
 | 
						|
#include "cpptoolsconstants.h"
 | 
						|
 | 
						|
#include <coreplugin/icore.h>
 | 
						|
 | 
						|
#include <utils/qtcassert.h>
 | 
						|
 | 
						|
using namespace CPlusPlus;
 | 
						|
using namespace CppTools;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
static int ordering(InsertionPointLocator::AccessSpec xsSpec)
 | 
						|
{
 | 
						|
    static QList<InsertionPointLocator::AccessSpec> order = QList<InsertionPointLocator::AccessSpec>()
 | 
						|
            << InsertionPointLocator::Public
 | 
						|
            << InsertionPointLocator::PublicSlot
 | 
						|
            << InsertionPointLocator::Signals
 | 
						|
            << InsertionPointLocator::Protected
 | 
						|
            << InsertionPointLocator::ProtectedSlot
 | 
						|
            << InsertionPointLocator::PrivateSlot
 | 
						|
            << InsertionPointLocator::Private
 | 
						|
            ;
 | 
						|
 | 
						|
    return order.indexOf(xsSpec);
 | 
						|
}
 | 
						|
 | 
						|
struct AccessRange
 | 
						|
{
 | 
						|
    unsigned start;
 | 
						|
    unsigned end;
 | 
						|
    InsertionPointLocator::AccessSpec xsSpec;
 | 
						|
    unsigned colonToken;
 | 
						|
 | 
						|
    AccessRange()
 | 
						|
        : start(0)
 | 
						|
        , end(0)
 | 
						|
        , xsSpec(InsertionPointLocator::Invalid)
 | 
						|
        , colonToken(0)
 | 
						|
    {}
 | 
						|
 | 
						|
    AccessRange(unsigned start, unsigned end, InsertionPointLocator::AccessSpec xsSpec, unsigned colonToken)
 | 
						|
        : start(start)
 | 
						|
        , end(end)
 | 
						|
        , xsSpec(xsSpec)
 | 
						|
        , colonToken(colonToken)
 | 
						|
    {}
 | 
						|
 | 
						|
    bool isEmpty() const
 | 
						|
    {
 | 
						|
        unsigned contentStart = 1 + (colonToken ? colonToken : start);
 | 
						|
        return contentStart == end;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
class FindInClass: public ASTVisitor
 | 
						|
{
 | 
						|
public:
 | 
						|
    FindInClass(const Document::Ptr &doc, const Class *clazz, InsertionPointLocator::AccessSpec xsSpec)
 | 
						|
        : ASTVisitor(doc->translationUnit())
 | 
						|
        , _doc(doc)
 | 
						|
        , _clazz(clazz)
 | 
						|
        , _xsSpec(xsSpec)
 | 
						|
    {}
 | 
						|
 | 
						|
    InsertionLocation operator()()
 | 
						|
    {
 | 
						|
        _result = InsertionLocation();
 | 
						|
 | 
						|
        AST *ast = translationUnit()->ast();
 | 
						|
        accept(ast);
 | 
						|
 | 
						|
        return _result;
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    using ASTVisitor::visit;
 | 
						|
 | 
						|
    bool visit(ClassSpecifierAST *ast)
 | 
						|
    {
 | 
						|
        if (!ast->lbrace_token || !ast->rbrace_token)
 | 
						|
            return true;
 | 
						|
        if (!ast->symbol || !ast->symbol->isEqualTo(_clazz))
 | 
						|
            return true;
 | 
						|
 | 
						|
        QList<AccessRange> ranges = collectAccessRanges(
 | 
						|
                    ast->member_specifier_list,
 | 
						|
                    tokenKind(ast->classkey_token) == T_CLASS ? InsertionPointLocator::Private : InsertionPointLocator::Public,
 | 
						|
                    ast->lbrace_token,
 | 
						|
                    ast->rbrace_token);
 | 
						|
 | 
						|
        unsigned beforeToken = 0;
 | 
						|
        bool needsLeadingEmptyLine = false;
 | 
						|
        bool needsPrefix = false;
 | 
						|
        bool needsSuffix = false;
 | 
						|
        findMatch(ranges, _xsSpec, beforeToken, needsLeadingEmptyLine, needsPrefix, needsSuffix);
 | 
						|
 | 
						|
        unsigned line = 0, column = 0;
 | 
						|
        getTokenStartPosition(beforeToken, &line, &column);
 | 
						|
 | 
						|
        QString prefix;
 | 
						|
        if (needsLeadingEmptyLine)
 | 
						|
            prefix += QLatin1String("\n");
 | 
						|
        if (needsPrefix)
 | 
						|
            prefix += InsertionPointLocator::accessSpecToString(_xsSpec);
 | 
						|
 | 
						|
        QString suffix;
 | 
						|
        if (needsSuffix)
 | 
						|
            suffix = QLatin1Char('\n');
 | 
						|
 | 
						|
        _result = InsertionLocation(_doc->fileName(), prefix, suffix,
 | 
						|
                                    line, column);
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    static void findMatch(const QList<AccessRange> &ranges,
 | 
						|
                          InsertionPointLocator::AccessSpec xsSpec,
 | 
						|
                          unsigned &beforeToken,
 | 
						|
                          bool &needsLeadingEmptyLine,
 | 
						|
                          bool &needsPrefix,
 | 
						|
                          bool &needsSuffix)
 | 
						|
    {
 | 
						|
        QTC_ASSERT(!ranges.isEmpty(), return);
 | 
						|
        const int lastIndex = ranges.size() - 1;
 | 
						|
 | 
						|
        needsLeadingEmptyLine = false;
 | 
						|
 | 
						|
        // try an exact match, and ignore the first (default) access spec:
 | 
						|
        for (int i = lastIndex; i > 0; --i) {
 | 
						|
            const AccessRange &range = ranges.at(i);
 | 
						|
            if (range.xsSpec == xsSpec) {
 | 
						|
                beforeToken = range.end;
 | 
						|
                needsPrefix = false;
 | 
						|
                needsSuffix = (i != lastIndex);
 | 
						|
                return;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // try to find a fitting access spec to insert XXX:
 | 
						|
        for (int i = lastIndex; i > 0; --i) {
 | 
						|
            const AccessRange ¤t = ranges.at(i);
 | 
						|
 | 
						|
            if (ordering(xsSpec) > ordering(current.xsSpec)) {
 | 
						|
                beforeToken = current.end;
 | 
						|
                needsPrefix = true;
 | 
						|
                needsSuffix = (i != lastIndex);
 | 
						|
                return;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // otherwise:
 | 
						|
        beforeToken = ranges.first().end;
 | 
						|
        needsLeadingEmptyLine = !ranges.first().isEmpty();
 | 
						|
        needsPrefix = true;
 | 
						|
        needsSuffix = (ranges.size() != 1);
 | 
						|
    }
 | 
						|
 | 
						|
    QList<AccessRange> collectAccessRanges(DeclarationListAST *decls,
 | 
						|
                                           InsertionPointLocator::AccessSpec initialXs,
 | 
						|
                                           int firstRangeStart,
 | 
						|
                                           int lastRangeEnd) const
 | 
						|
    {
 | 
						|
        QList<AccessRange> ranges;
 | 
						|
        ranges.append(AccessRange(firstRangeStart, lastRangeEnd, initialXs, 0));
 | 
						|
 | 
						|
        for (DeclarationListAST *iter = decls; iter; iter = iter->next) {
 | 
						|
            DeclarationAST *decl = iter->value;
 | 
						|
 | 
						|
            if (AccessDeclarationAST *xsDecl = decl->asAccessDeclaration()) {
 | 
						|
                const unsigned token = xsDecl->access_specifier_token;
 | 
						|
                InsertionPointLocator::AccessSpec newXsSpec = initialXs;
 | 
						|
                bool isSlot = xsDecl->slots_token
 | 
						|
                        && tokenKind(xsDecl->slots_token) == T_Q_SLOTS;
 | 
						|
 | 
						|
                switch (tokenKind(token)) {
 | 
						|
                case T_PUBLIC:
 | 
						|
                    newXsSpec = isSlot ? InsertionPointLocator::PublicSlot
 | 
						|
                                       : InsertionPointLocator::Public;
 | 
						|
                    break;
 | 
						|
 | 
						|
                case T_PROTECTED:
 | 
						|
                    newXsSpec = isSlot ? InsertionPointLocator::ProtectedSlot
 | 
						|
                                       : InsertionPointLocator::Protected;
 | 
						|
                    break;
 | 
						|
 | 
						|
                case T_PRIVATE:
 | 
						|
                    newXsSpec = isSlot ? InsertionPointLocator::PrivateSlot
 | 
						|
                                       : InsertionPointLocator::Private;
 | 
						|
                    break;
 | 
						|
 | 
						|
                case T_Q_SIGNALS:
 | 
						|
                    newXsSpec = InsertionPointLocator::Signals;
 | 
						|
                    break;
 | 
						|
 | 
						|
                case T_Q_SLOTS: {
 | 
						|
                    newXsSpec = (InsertionPointLocator::AccessSpec)
 | 
						|
                            (ranges.last().xsSpec | InsertionPointLocator::SlotBit);
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
 | 
						|
                default:
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
 | 
						|
                if (newXsSpec != ranges.last().xsSpec || ranges.size() == 1) {
 | 
						|
                    ranges.last().end = token;
 | 
						|
                    AccessRange r(token, lastRangeEnd, newXsSpec, xsDecl->colon_token);
 | 
						|
                    ranges.append(r);
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        ranges.last().end = lastRangeEnd;
 | 
						|
        return ranges;
 | 
						|
    }
 | 
						|
 | 
						|
private:
 | 
						|
    Document::Ptr _doc;
 | 
						|
    const Class *_clazz;
 | 
						|
    InsertionPointLocator::AccessSpec _xsSpec;
 | 
						|
 | 
						|
    InsertionLocation _result;
 | 
						|
};
 | 
						|
 | 
						|
} // end of anonymous namespace
 | 
						|
 | 
						|
InsertionLocation::InsertionLocation()
 | 
						|
    : m_line(0)
 | 
						|
    , m_column(0)
 | 
						|
{}
 | 
						|
 | 
						|
InsertionLocation::InsertionLocation(const QString &fileName,
 | 
						|
                                     const QString &prefix,
 | 
						|
                                     const QString &suffix,
 | 
						|
                                     unsigned line, unsigned column)
 | 
						|
    : m_fileName(fileName)
 | 
						|
    , m_prefix(prefix)
 | 
						|
    , m_suffix(suffix)
 | 
						|
    , m_line(line)
 | 
						|
    , m_column(column)
 | 
						|
{}
 | 
						|
 | 
						|
QString InsertionPointLocator::accessSpecToString(InsertionPointLocator::AccessSpec xsSpec)
 | 
						|
{
 | 
						|
    switch (xsSpec) {
 | 
						|
    default:
 | 
						|
    case InsertionPointLocator::Public:
 | 
						|
        return QLatin1String("public:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::Protected:
 | 
						|
        return QLatin1String("protected:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::Private:
 | 
						|
        return QLatin1String("private:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::PublicSlot:
 | 
						|
        return QLatin1String("public slots:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::ProtectedSlot:
 | 
						|
        return QLatin1String("protected slots:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::PrivateSlot:
 | 
						|
        return QLatin1String("private slots:\n");
 | 
						|
 | 
						|
    case InsertionPointLocator::Signals:
 | 
						|
        return QLatin1String("signals:\n");
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
InsertionPointLocator::InsertionPointLocator(const CppRefactoringChanges &refactoringChanges)
 | 
						|
    : m_refactoringChanges(refactoringChanges)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
InsertionLocation InsertionPointLocator::methodDeclarationInClass(
 | 
						|
    const QString &fileName,
 | 
						|
    const Class *clazz,
 | 
						|
    AccessSpec xsSpec) const
 | 
						|
{
 | 
						|
    const Document::Ptr doc = m_refactoringChanges.file(fileName)->cppDocument();
 | 
						|
    if (doc) {
 | 
						|
        FindInClass find(doc, clazz, xsSpec);
 | 
						|
        return find();
 | 
						|
    } else {
 | 
						|
        return InsertionLocation();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
namespace {
 | 
						|
template <class Key, class Value>
 | 
						|
class HighestValue
 | 
						|
{
 | 
						|
    Key _key;
 | 
						|
    Value _value;
 | 
						|
    bool _set;
 | 
						|
public:
 | 
						|
    HighestValue()
 | 
						|
        : _key(), _set(false)
 | 
						|
    {}
 | 
						|
 | 
						|
    HighestValue(const Key &initialKey, const Value &initialValue)
 | 
						|
        : _key(initialKey)
 | 
						|
        , _value(initialValue)
 | 
						|
        , _set(true)
 | 
						|
    {}
 | 
						|
 | 
						|
    void maybeSet(const Key &key, const Value &value)
 | 
						|
    {
 | 
						|
        if (!_set || key > _key) {
 | 
						|
            _value = value;
 | 
						|
            _key = key;
 | 
						|
            _set = true;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    const Value &get() const
 | 
						|
    {
 | 
						|
        QTC_CHECK(_set);
 | 
						|
        return _value;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
class FindMethodDefinitionInsertPoint : protected ASTVisitor
 | 
						|
{
 | 
						|
    QList<const Identifier *> _namespaceNames;
 | 
						|
    int _currentDepth;
 | 
						|
    HighestValue<int, unsigned> _bestToken;
 | 
						|
 | 
						|
public:
 | 
						|
    FindMethodDefinitionInsertPoint(TranslationUnit *translationUnit)
 | 
						|
        : ASTVisitor(translationUnit)
 | 
						|
    {}
 | 
						|
 | 
						|
    void operator()(Symbol *decl, unsigned *line, unsigned *column)
 | 
						|
    {
 | 
						|
        // default to end of file
 | 
						|
        const unsigned lastToken = translationUnit()->ast()->lastToken();
 | 
						|
        _bestToken.maybeSet(-1, lastToken);
 | 
						|
 | 
						|
        if (lastToken >= 2) {
 | 
						|
            QList<const Name *> names = LookupContext::fullyQualifiedName(decl);
 | 
						|
            foreach (const Name *name, names) {
 | 
						|
                const Identifier *id = name->asNameId();
 | 
						|
                if (!id)
 | 
						|
                    break;
 | 
						|
                _namespaceNames += id;
 | 
						|
            }
 | 
						|
            _currentDepth = 0;
 | 
						|
 | 
						|
            accept(translationUnit()->ast());
 | 
						|
        }
 | 
						|
 | 
						|
        if (lastToken == _bestToken.get()) // No matching namespace found
 | 
						|
            translationUnit()->getTokenStartPosition(lastToken, line, column);
 | 
						|
        else // Insert at end of matching namespace
 | 
						|
            translationUnit()->getTokenEndPosition(_bestToken.get(), line, column);
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    bool preVisit(AST *ast)
 | 
						|
    {
 | 
						|
        return ast->asNamespace() || ast->asTranslationUnit() || ast->asLinkageBody();
 | 
						|
    }
 | 
						|
 | 
						|
    bool visit(NamespaceAST *ast)
 | 
						|
    {
 | 
						|
        if (_currentDepth >= _namespaceNames.size())
 | 
						|
            return false;
 | 
						|
 | 
						|
        // ignore anonymous namespaces
 | 
						|
        if (!ast->identifier_token)
 | 
						|
            return false;
 | 
						|
 | 
						|
        const Identifier *name = translationUnit()->identifier(ast->identifier_token);
 | 
						|
        if (!name->equalTo(_namespaceNames.at(_currentDepth)))
 | 
						|
            return false;
 | 
						|
 | 
						|
        // found a good namespace
 | 
						|
        _bestToken.maybeSet(_currentDepth, ast->lastToken() - 2);
 | 
						|
 | 
						|
        ++_currentDepth;
 | 
						|
        accept(ast->linkage_body);
 | 
						|
        --_currentDepth;
 | 
						|
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
class FindFunctionDefinition : protected ASTVisitor
 | 
						|
{
 | 
						|
    FunctionDefinitionAST *_result;
 | 
						|
    unsigned _line, _column;
 | 
						|
public:
 | 
						|
    FindFunctionDefinition(TranslationUnit *translationUnit)
 | 
						|
        : ASTVisitor(translationUnit)
 | 
						|
    {
 | 
						|
    }
 | 
						|
 | 
						|
    FunctionDefinitionAST *operator()(unsigned line, unsigned column)
 | 
						|
    {
 | 
						|
        _result = 0;
 | 
						|
        _line = line;
 | 
						|
        _column = column;
 | 
						|
        accept(translationUnit()->ast());
 | 
						|
        return _result;
 | 
						|
    }
 | 
						|
 | 
						|
protected:
 | 
						|
    bool preVisit(AST *ast)
 | 
						|
    {
 | 
						|
        if (_result)
 | 
						|
            return false;
 | 
						|
        unsigned line, column;
 | 
						|
        translationUnit()->getTokenStartPosition(ast->firstToken(), &line, &column);
 | 
						|
        if (line > _line || (line == _line && column > _column))
 | 
						|
            return false;
 | 
						|
        translationUnit()->getTokenEndPosition(ast->lastToken() - 1, &line, &column);
 | 
						|
        if (line < _line || (line == _line && column < _column))
 | 
						|
            return false;
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    bool visit(FunctionDefinitionAST *ast)
 | 
						|
    {
 | 
						|
        _result = ast;
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
} // anonymous namespace
 | 
						|
 | 
						|
static Declaration *isNonVirtualFunctionDeclaration(Symbol *s)
 | 
						|
{
 | 
						|
    if (!s)
 | 
						|
        return 0;
 | 
						|
    Declaration *declaration = s->asDeclaration();
 | 
						|
    if (!declaration)
 | 
						|
        return 0;
 | 
						|
    Function *type = s->type()->asFunctionType();
 | 
						|
    if (!type || type->isPureVirtual())
 | 
						|
        return 0;
 | 
						|
    return declaration;
 | 
						|
}
 | 
						|
 | 
						|
static InsertionLocation nextToSurroundingDefinitions(Symbol *declaration,
 | 
						|
                                                      const CppRefactoringChanges &changes,
 | 
						|
                                                      const QString& destinationFile)
 | 
						|
{
 | 
						|
    InsertionLocation noResult;
 | 
						|
    Class *klass = declaration->enclosingClass();
 | 
						|
    if (!klass)
 | 
						|
        return noResult;
 | 
						|
 | 
						|
    // find the index of declaration
 | 
						|
    int declIndex = -1;
 | 
						|
    for (unsigned i = 0; i < klass->memberCount(); ++i) {
 | 
						|
        Symbol *s = klass->memberAt(i);
 | 
						|
        if (s == declaration) {
 | 
						|
            declIndex = i;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    if (declIndex == -1)
 | 
						|
        return noResult;
 | 
						|
 | 
						|
    // scan preceding declarations for a function declaration (and see if it is defined)
 | 
						|
    CppTools::SymbolFinder symbolFinder;
 | 
						|
    Function *definitionFunction = 0;
 | 
						|
    QString prefix, suffix;
 | 
						|
    Declaration *surroundingFunctionDecl = 0;
 | 
						|
    for (int i = declIndex - 1; i >= 0; --i) {
 | 
						|
        Symbol *s = klass->memberAt(i);
 | 
						|
        if (s->isGenerated() || !(surroundingFunctionDecl = isNonVirtualFunctionDeclaration(s)))
 | 
						|
            continue;
 | 
						|
        if ((definitionFunction = symbolFinder.findMatchingDefinition(surroundingFunctionDecl,
 | 
						|
                                                                      changes.snapshot())))
 | 
						|
        {
 | 
						|
            if (destinationFile.isEmpty() || destinationFile == QString::fromUtf8(
 | 
						|
                        definitionFunction->fileName(), definitionFunction->fileNameLength())) {
 | 
						|
                prefix = QLatin1String("\n\n");
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            definitionFunction = 0;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    if (!definitionFunction) {
 | 
						|
        // try to find one below
 | 
						|
        for (unsigned i = declIndex + 1; i < klass->memberCount(); ++i) {
 | 
						|
            Symbol *s = klass->memberAt(i);
 | 
						|
            surroundingFunctionDecl = isNonVirtualFunctionDeclaration(s);
 | 
						|
            if (!surroundingFunctionDecl)
 | 
						|
                continue;
 | 
						|
            if ((definitionFunction = symbolFinder.findMatchingDefinition(surroundingFunctionDecl,
 | 
						|
                                                                          changes.snapshot())))
 | 
						|
            {
 | 
						|
                if (destinationFile.isEmpty() || destinationFile == QString::fromUtf8(
 | 
						|
                            definitionFunction->fileName(), definitionFunction->fileNameLength())) {
 | 
						|
                    suffix = QLatin1String("\n\n");
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
                definitionFunction = 0;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!definitionFunction)
 | 
						|
        return noResult;
 | 
						|
 | 
						|
    unsigned line, column;
 | 
						|
    if (suffix.isEmpty()) {
 | 
						|
        Document::Ptr targetDoc = changes.snapshot().document(QString::fromUtf8(definitionFunction->fileName()));
 | 
						|
        if (!targetDoc)
 | 
						|
            return noResult;
 | 
						|
 | 
						|
        targetDoc->translationUnit()->getPosition(definitionFunction->endOffset(), &line, &column);
 | 
						|
    } else {
 | 
						|
        // we don't have an offset to the start of the function definition, so we need to manually find it...
 | 
						|
        CppRefactoringFilePtr targetFile = changes.file(QString::fromUtf8(definitionFunction->fileName()));
 | 
						|
        if (!targetFile->isValid())
 | 
						|
            return noResult;
 | 
						|
 | 
						|
        FindFunctionDefinition finder(targetFile->cppDocument()->translationUnit());
 | 
						|
        FunctionDefinitionAST *functionDefinition = finder(definitionFunction->line(), definitionFunction->column());
 | 
						|
        if (!functionDefinition)
 | 
						|
            return noResult;
 | 
						|
 | 
						|
        targetFile->cppDocument()->translationUnit()->getTokenStartPosition(functionDefinition->firstToken(), &line, &column);
 | 
						|
    }
 | 
						|
 | 
						|
    return InsertionLocation(QString::fromUtf8(definitionFunction->fileName()), prefix, suffix, line, column);
 | 
						|
}
 | 
						|
 | 
						|
QList<InsertionLocation> InsertionPointLocator::methodDefinition(Symbol *declaration,
 | 
						|
                                                                 bool useSymbolFinder,
 | 
						|
                                                                 const QString &destinationFile) const
 | 
						|
{
 | 
						|
    QList<InsertionLocation> result;
 | 
						|
    if (!declaration)
 | 
						|
        return result;
 | 
						|
 | 
						|
    if (useSymbolFinder) {
 | 
						|
        CppTools::SymbolFinder symbolFinder;
 | 
						|
        if (symbolFinder.findMatchingDefinition(declaration, m_refactoringChanges.snapshot(), true))
 | 
						|
            return result;
 | 
						|
    }
 | 
						|
 | 
						|
    const InsertionLocation location = nextToSurroundingDefinitions(declaration,
 | 
						|
                                                                    m_refactoringChanges,
 | 
						|
                                                                    destinationFile);
 | 
						|
    if (location.isValid()) {
 | 
						|
        result += location;
 | 
						|
        return result;
 | 
						|
    }
 | 
						|
 | 
						|
    const QString declFileName = QString::fromUtf8(declaration->fileName(),
 | 
						|
                                                   declaration->fileNameLength());
 | 
						|
    QString target = declFileName;
 | 
						|
    if (!ProjectFile::isSource(ProjectFile::classify(declFileName))) {
 | 
						|
        QString candidate = CppTools::correspondingHeaderOrSource(declFileName);
 | 
						|
        if (!candidate.isEmpty())
 | 
						|
            target = candidate;
 | 
						|
    }
 | 
						|
 | 
						|
    CppRefactoringFilePtr targetFile = m_refactoringChanges.file(target);
 | 
						|
    Document::Ptr doc = targetFile->cppDocument();
 | 
						|
    if (doc.isNull())
 | 
						|
        return result;
 | 
						|
 | 
						|
    unsigned line = 0, column = 0;
 | 
						|
    FindMethodDefinitionInsertPoint finder(doc->translationUnit());
 | 
						|
    finder(declaration, &line, &column);
 | 
						|
 | 
						|
    // Force empty lines before and after the new definition.
 | 
						|
    QString prefix;
 | 
						|
    QString suffix;
 | 
						|
    if (!line) {
 | 
						|
        // Totally empty file.
 | 
						|
        line = 1;
 | 
						|
        column = 1;
 | 
						|
        prefix = suffix = QLatin1Char('\n');
 | 
						|
    } else {
 | 
						|
        QTC_ASSERT(column, return result);
 | 
						|
 | 
						|
        prefix = QLatin1String("\n\n");
 | 
						|
        int firstNonSpace = targetFile->position(line, column);
 | 
						|
        QChar c = targetFile->charAt(firstNonSpace);
 | 
						|
        while (c == QLatin1Char(' ') || c == QLatin1Char('\t')) {
 | 
						|
            ++firstNonSpace;
 | 
						|
            c = targetFile->charAt(firstNonSpace);
 | 
						|
        }
 | 
						|
        if (targetFile->charAt(firstNonSpace) != QChar::ParagraphSeparator) {
 | 
						|
            suffix.append(QLatin1String("\n\n"));
 | 
						|
        } else {
 | 
						|
            ++firstNonSpace;
 | 
						|
            if (targetFile->charAt(firstNonSpace) != QChar::ParagraphSeparator)
 | 
						|
                suffix.append(QLatin1Char('\n'));
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    result += InsertionLocation(target, prefix, suffix, line, column);
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 |