/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** Commercial Usage ** ** Licensees holding valid Qt Commercial licenses may use this file in ** accordance with the Qt Commercial License Agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Nokia. ** ** 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. ** ** If you are unsure which license is appropriate for your use, please ** contact the sales department at http://qt.nokia.com/contact. ** **************************************************************************/ #include "cppdeclfromdef.h" #include <Literals.h> //### remove #include <QDebug> //###remove #include <AST.h> #include <ASTVisitor.h> #include <CoreTypes.h> #include <Names.h> #include <Symbols.h> #include <TranslationUnit.h> #include <cplusplus/ASTPath.h> #include <cplusplus/InsertionPointLocator.h> #include <cplusplus/LookupContext.h> #include <cplusplus/Overview.h> #include <QtCore/QCoreApplication> using namespace CPlusPlus; using namespace CppEditor; using namespace CppEditor::Internal; using namespace CppTools; using CppEditor::CppRefactoringChanges; namespace { class Operation: public CppQuickFixOperation { public: Operation(const CppQuickFixState &state, int priority, const QString &targetFileName, const Class *targetSymbol, InsertionPointLocator::AccessSpec xsSpec, const QString &decl) : CppQuickFixOperation(state, priority) , m_targetFileName(targetFileName) , m_targetSymbol(targetSymbol) , m_xsSpec(xsSpec) , m_decl(decl) { QString type; switch (xsSpec) { case InsertionPointLocator::Public: type = QLatin1String("public"); break; case InsertionPointLocator::Protected: type = QLatin1String("protected"); break; case InsertionPointLocator::Private: type = QLatin1String("private"); break; case InsertionPointLocator::PublicSlot: type = QLatin1String("public slot"); break; case InsertionPointLocator::ProtectedSlot: type = QLatin1String("protected slot"); break; case InsertionPointLocator::PrivateSlot: type = QLatin1String("private slot"); break; default: break; } setDescription(QCoreApplication::tr("Create %1 Declaration from Definition", "CppEditor::DeclFromDef").arg(type)); } void performChanges(CppRefactoringFile *, CppRefactoringChanges *refactoring) { CppRefactoringFile targetFile = refactoring->file(m_targetFileName); Document::Ptr targetDoc = targetFile.cppDocument(); InsertionPointLocator locator(targetDoc); const InsertionLocation loc = locator.methodDeclarationInClass(m_targetSymbol, m_xsSpec); Q_ASSERT(loc.isValid()); int targetPosition1 = targetFile.position(loc.line(), loc.column()); int targetPosition2 = qMax(0, targetFile.position(loc.line(), 1) - 1); Utils::ChangeSet target; target.insert(targetPosition1, loc.prefix() + m_decl); targetFile.change(target); targetFile.indent(Utils::ChangeSet::Range(targetPosition2, targetPosition1)); refactoring->activateEditor(m_targetFileName, targetPosition1); } private: QString m_targetFileName; const Class *m_targetSymbol; InsertionPointLocator::AccessSpec m_xsSpec; QString m_decl; }; } // anonymous namespace QList<CppQuickFixOperation::Ptr> DeclFromDef::match(const CppQuickFixState &state) { const QList<AST *> &path = state.path(); const CppRefactoringFile &file = state.currentFile(); FunctionDefinitionAST *funDef = 0; int idx = 0; for (; idx < path.size(); ++idx) { AST *node = path.at(idx); if (FunctionDefinitionAST *candidate = node->asFunctionDefinition()) { if (!funDef && file.isCursorOn(candidate) && !file.isCursorOn(candidate->function_body)) funDef = candidate; } else if (node->asClassSpecifier()) { return noResult(); } } if (!funDef || !funDef->symbol) return noResult(); Function *method = funDef->symbol; if (ClassOrNamespace *targetBinding = state.context().lookupParent(method)) { foreach (Symbol *s, targetBinding->symbols()) { if (Class *clazz = s->asClass()) { QList<CppQuickFixOperation::Ptr> results; const QLatin1String fn(clazz->fileName()); const QString decl = generateDeclaration(state, method, targetBinding); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::Public, decl))); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::Protected, decl))); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::Private, decl))); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::PublicSlot, decl))); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::ProtectedSlot, decl))); results.append(singleResult(new Operation(state, idx, fn, clazz, InsertionPointLocator::PrivateSlot, decl))); return results; } //! \todo support insertion into namespaces } } return noResult(); } QString DeclFromDef::generateDeclaration(const CppQuickFixState &, Function *method, ClassOrNamespace *targetBinding) { Q_UNUSED(targetBinding); Overview oo; oo.setShowFunctionSignatures(true); oo.setShowReturnTypes(true); oo.setShowArgumentNames(true); QString decl; decl += oo(method->type(), method->unqualifiedName()); decl += QLatin1String(";\n"); return decl; }