| 
									
										
										
										
											2012-10-02 09:12:39 +02:00
										 |  |  | /****************************************************************************
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2013-01-28 17:12:19 +01:00
										 |  |  | ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2012-10-02 09:12:39 +02:00
										 |  |  | ** Contact: http://www.qt-project.org/legal
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2012-10-02 09:12:39 +02:00
										 |  |  | ** This file is part of Qt Creator. | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +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 | 
					
						
							|  |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +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 | 
					
						
							|  |  |  | ** 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 | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2012-10-02 09:12:39 +02:00
										 |  |  | ****************************************************************************/ | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | #include "cppquickfixes.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 14:32:42 +01:00
										 |  |  | #include "cppcompleteswitch.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | #include "cppeditor.h"
 | 
					
						
							| 
									
										
										
										
											2010-09-02 15:17:57 +02:00
										 |  |  | #include "cppinsertdecldef.h"
 | 
					
						
							| 
									
										
										
										
											2010-12-10 15:42:34 +01:00
										 |  |  | #include "cppinsertqtpropertymembers.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-15 16:19:23 +02:00
										 |  |  | #include "cppquickfixassistant.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | #include <cpptools/cppclassesfilter.h>
 | 
					
						
							| 
									
										
										
										
											2013-01-15 14:26:24 +01:00
										 |  |  | #include <cpptools/cppcodestylesettings.h>
 | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | #include <cpptools/cpppointerdeclarationformatter.h>
 | 
					
						
							|  |  |  | #include <cpptools/cpptoolsconstants.h>
 | 
					
						
							| 
									
										
										
										
											2012-01-20 14:43:21 +01:00
										 |  |  | #include <cpptools/symbolfinder.h>
 | 
					
						
							| 
									
										
										
										
											2013-03-27 18:54:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <cplusplus/CppRewriter.h>
 | 
					
						
							|  |  |  | #include <cplusplus/DependencyTable.h>
 | 
					
						
							|  |  |  | #include <cplusplus/TypeOfExpression.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | #include <extensionsystem/pluginmanager.h>
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | #include <utils/qtcassert.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-15 10:42:41 +01:00
										 |  |  | #include <QApplication>
 | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | #include <QFileInfo>
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | #include <QSharedPointer>
 | 
					
						
							| 
									
										
										
										
											2012-02-15 10:42:41 +01:00
										 |  |  | #include <QTextBlock>
 | 
					
						
							|  |  |  | #include <QTextCursor>
 | 
					
						
							| 
									
										
										
										
											2011-10-25 14:06:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-15 08:26:12 +01:00
										 |  |  | #include <cctype>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | using namespace CppEditor; | 
					
						
							|  |  |  | using namespace CppEditor::Internal; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void CppEditor::Internal::registerQuickFixes(ExtensionSystem::IPlugin *plugIn) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new AddIncludeForUndefinedIdentifier); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new AddIncludeForForwardDeclaration); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new FlipLogicalOperands); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new InverseLogicalComparison); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new RewriteLogicalAnd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ConvertToCamelCase); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ConvertCStringToNSString); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ConvertNumericLiteral); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new TranslateStringLiteral); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new WrapStringLiteral); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new MoveDeclarationOutOfIf); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new MoveDeclarationOutOfWhile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new SplitIfStatement); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new SplitSimpleDeclaration); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new AddLocalDeclaration); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new AddBracesToIf); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new RearrangeParamDeclarationList); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ReformatPointerDeclaration); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new CompleteSwitchCaseStatement); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new InsertQtPropertyMembers); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ApplyDeclDefLinkChanges); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new ExtractFunction); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new GenerateGetterSetter); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new InsertDeclFromDef); | 
					
						
							|  |  |  |     plugIn->addAutoReleasedObject(new InsertDefFromDecl); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-30 10:45:11 +02:00
										 |  |  | static inline bool isQtStringLiteral(const QByteArray &id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return id == "QLatin1String" || id == "QLatin1Literal" || id == "QStringLiteral"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | static inline bool isQtStringTranslation(const QByteArray &id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return id == "tr" || id == "trUtf8" || id == "translate" || id == "QT_TRANSLATE_NOOP"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class InverseLogicalComparisonOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     InverseLogicalComparisonOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                                BinaryExpressionAST *binary, Kind invertToken) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , binary(binary), nested(0), negation(0) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         Token tok; | 
					
						
							|  |  |  |         tok.f.kind = invertToken; | 
					
						
							|  |  |  |         replacement = QLatin1String(tok.spell()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         // check for enclosing nested expression
 | 
					
						
							|  |  |  |         if (priority - 1 >= 0) | 
					
						
							|  |  |  |             nested = interface->path()[priority - 1]->asNestedExpression(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         // check for ! before parentheses
 | 
					
						
							|  |  |  |         if (nested && priority - 2 >= 0) { | 
					
						
							|  |  |  |             negation = interface->path()[priority - 2]->asUnaryExpression(); | 
					
						
							|  |  |  |             if (negation && ! interface->currentFile()->tokenAt(negation->unary_op_token).is(T_EXCLAIM)) | 
					
						
							|  |  |  |                 negation = 0; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString description() const | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         return QApplication::translate("CppTools::QuickFix", "Rewrite Using %1").arg(replacement); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         if (negation) { | 
					
						
							|  |  |  |             // can't remove parentheses since that might break precedence
 | 
					
						
							|  |  |  |             changes.remove(currentFile->range(negation->unary_op_token)); | 
					
						
							|  |  |  |         } else if (nested) { | 
					
						
							|  |  |  |             changes.insert(currentFile->startOf(nested), QLatin1String("!")); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             changes.insert(currentFile->startOf(binary), QLatin1String("!(")); | 
					
						
							|  |  |  |             changes.insert(currentFile->endOf(binary), QLatin1String(")")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         changes.replace(currentFile->range(binary->binary_op_token), replacement); | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     BinaryExpressionAST *binary; | 
					
						
							|  |  |  |     NestedExpressionAST *nested; | 
					
						
							|  |  |  |     UnaryExpressionAST *negation; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString replacement; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void InverseLogicalComparison::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                      QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     BinaryExpressionAST *binary = path.at(index)->asBinaryExpression(); | 
					
						
							|  |  |  |     if (! binary) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (! interface->isCursorOn(binary->binary_op_token)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Kind invertToken; | 
					
						
							|  |  |  |     switch (file->tokenAt(binary->binary_op_token).kind()) { | 
					
						
							|  |  |  |     case T_LESS_EQUAL: | 
					
						
							|  |  |  |         invertToken = T_GREATER; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_LESS: | 
					
						
							|  |  |  |         invertToken = T_GREATER_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_GREATER: | 
					
						
							|  |  |  |         invertToken = T_LESS_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_GREATER_EQUAL: | 
					
						
							|  |  |  |         invertToken = T_LESS; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_EQUAL_EQUAL: | 
					
						
							|  |  |  |         invertToken = T_EXCLAIM_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_EXCLAIM_EQUAL: | 
					
						
							|  |  |  |         invertToken = T_EQUAL_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     result.append(CppQuickFixOperation::Ptr( | 
					
						
							|  |  |  |         new InverseLogicalComparisonOp(interface, index, binary, invertToken))); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class FlipLogicalOperandsOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     FlipLogicalOperandsOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                           BinaryExpressionAST *binary, QString replacement) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |         , binary(binary) | 
					
						
							|  |  |  |         , replacement(replacement) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setPriority(priority); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString description() const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (replacement.isEmpty()) | 
					
						
							|  |  |  |             return QApplication::translate("CppTools::QuickFix", "Swap Operands"); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             return QApplication::translate("CppTools::QuickFix", "Rewrite Using %1").arg(replacement); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         changes.flip(currentFile->range(binary->left_expression), currentFile->range(binary->right_expression)); | 
					
						
							|  |  |  |         if (! replacement.isEmpty()) | 
					
						
							|  |  |  |             changes.replace(currentFile->range(binary->binary_op_token), replacement); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     BinaryExpressionAST *binary; | 
					
						
							|  |  |  |     QString replacement; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void FlipLogicalOperands::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     BinaryExpressionAST *binary = path.at(index)->asBinaryExpression(); | 
					
						
							|  |  |  |     if (! binary) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (! interface->isCursorOn(binary->binary_op_token)) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     Kind flipToken; | 
					
						
							|  |  |  |     switch (file->tokenAt(binary->binary_op_token).kind()) { | 
					
						
							|  |  |  |     case T_LESS_EQUAL: | 
					
						
							|  |  |  |         flipToken = T_GREATER_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_LESS: | 
					
						
							|  |  |  |         flipToken = T_GREATER; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_GREATER: | 
					
						
							|  |  |  |         flipToken = T_LESS; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_GREATER_EQUAL: | 
					
						
							|  |  |  |         flipToken = T_LESS_EQUAL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case T_EQUAL_EQUAL: | 
					
						
							|  |  |  |     case T_EXCLAIM_EQUAL: | 
					
						
							|  |  |  |     case T_AMPER_AMPER: | 
					
						
							|  |  |  |     case T_PIPE_PIPE: | 
					
						
							|  |  |  |         flipToken = T_EOF_SYMBOL; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString replacement; | 
					
						
							|  |  |  |     if (flipToken != T_EOF_SYMBOL) { | 
					
						
							|  |  |  |         Token tok; | 
					
						
							|  |  |  |         tok.f.kind = flipToken; | 
					
						
							|  |  |  |         replacement = QLatin1String(tok.spell()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |         new FlipLogicalOperandsOp(interface, index, binary, replacement))); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class RewriteLogicalAndOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QSharedPointer<ASTPatternBuilder> mk; | 
					
						
							|  |  |  |     UnaryExpressionAST *left; | 
					
						
							|  |  |  |     UnaryExpressionAST *right; | 
					
						
							|  |  |  |     BinaryExpressionAST *pattern; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     RewriteLogicalAndOp(const CppQuickFixInterface &interface) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |         , mk(new ASTPatternBuilder) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         left = mk->UnaryExpression(); | 
					
						
							|  |  |  |         right = mk->UnaryExpression(); | 
					
						
							|  |  |  |         pattern = mk->BinaryExpression(left, right); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         changes.replace(currentFile->range(pattern->binary_op_token), QLatin1String("||")); | 
					
						
							|  |  |  |         changes.remove(currentFile->range(left->unary_op_token)); | 
					
						
							|  |  |  |         changes.remove(currentFile->range(right->unary_op_token)); | 
					
						
							|  |  |  |         const int start = currentFile->startOf(pattern); | 
					
						
							|  |  |  |         const int end = currentFile->endOf(pattern); | 
					
						
							|  |  |  |         changes.insert(start, QLatin1String("!(")); | 
					
						
							|  |  |  |         changes.insert(end, QLatin1String(")")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(pattern)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void RewriteLogicalAnd::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     BinaryExpressionAST *expression = 0; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							|  |  |  |         expression = path.at(index)->asBinaryExpression(); | 
					
						
							|  |  |  |         if (expression) | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! expression) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! interface->isCursorOn(expression->binary_op_token)) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QSharedPointer<RewriteLogicalAndOp> op(new RewriteLogicalAndOp(interface)); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (expression->match(op->pattern, &matcher) && | 
					
						
							|  |  |  |             file->tokenAt(op->pattern->binary_op_token).is(T_AMPER_AMPER) && | 
					
						
							|  |  |  |             file->tokenAt(op->left->unary_op_token).is(T_EXCLAIM) && | 
					
						
							|  |  |  |             file->tokenAt(op->right->unary_op_token).is(T_EXCLAIM)) { | 
					
						
							|  |  |  |         op->setDescription(QApplication::translate("CppTools::QuickFix", "Rewrite Condition Using ||")); | 
					
						
							|  |  |  |         op->setPriority(index); | 
					
						
							|  |  |  |         result.append(op); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | bool SplitSimpleDeclaration::checkDeclaration(SimpleDeclarationAST *declaration) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! declaration->semicolon_token) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! declaration->decl_specifier_list) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2011-01-07 14:49:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     for (SpecifierListAST *it = declaration->decl_specifier_list; it; it = it->next) { | 
					
						
							|  |  |  |         SpecifierAST *specifier = it->value; | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (specifier->asEnumSpecifier() != 0) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         else if (specifier->asClassSpecifier() != 0) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! declaration->declarator_list) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     else if (! declaration->declarator_list->next) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class SplitSimpleDeclarationOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     SplitSimpleDeclarationOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                              SimpleDeclarationAST *decl) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , declaration(decl) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Split Declaration")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         SpecifierListAST *specifiers = declaration->decl_specifier_list; | 
					
						
							|  |  |  |         int declSpecifiersStart = currentFile->startOf(specifiers->firstToken()); | 
					
						
							|  |  |  |         int declSpecifiersEnd = currentFile->endOf(specifiers->lastToken() - 1); | 
					
						
							|  |  |  |         int insertPos = currentFile->endOf(declaration->semicolon_token); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         DeclaratorAST *prevDeclarator = declaration->declarator_list->value; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         for (DeclaratorListAST *it = declaration->declarator_list->next; it; it = it->next) { | 
					
						
							|  |  |  |             DeclaratorAST *declarator = it->value; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             changes.insert(insertPos, QLatin1String("\n")); | 
					
						
							|  |  |  |             changes.copy(declSpecifiersStart, declSpecifiersEnd, insertPos); | 
					
						
							|  |  |  |             changes.insert(insertPos, QLatin1String(" ")); | 
					
						
							|  |  |  |             changes.move(currentFile->range(declarator), insertPos); | 
					
						
							|  |  |  |             changes.insert(insertPos, QLatin1String(";")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             const int prevDeclEnd = currentFile->endOf(prevDeclarator); | 
					
						
							|  |  |  |             changes.remove(prevDeclEnd, currentFile->startOf(declarator)); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             prevDeclarator = declarator; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(declaration)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     SimpleDeclarationAST *declaration; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void SplitSimpleDeclaration::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                    QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoreDeclaratorAST *core_declarator = 0; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							|  |  |  |     const int cursorPosition = file->cursor().selectionStart(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     for (int index = path.size() - 1; index != -1; --index) { | 
					
						
							|  |  |  |         AST *node = path.at(index); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (CoreDeclaratorAST *coreDecl = node->asCoreDeclarator()) | 
					
						
							|  |  |  |             core_declarator = coreDecl; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         else if (SimpleDeclarationAST *simpleDecl = node->asSimpleDeclaration()) { | 
					
						
							|  |  |  |             if (checkDeclaration(simpleDecl)) { | 
					
						
							|  |  |  |                 SimpleDeclarationAST *declaration = simpleDecl; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 const int startOfDeclSpecifier = file->startOf(declaration->decl_specifier_list->firstToken()); | 
					
						
							|  |  |  |                 const int endOfDeclSpecifier = file->endOf(declaration->decl_specifier_list->lastToken() - 1); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (cursorPosition >= startOfDeclSpecifier && cursorPosition <= endOfDeclSpecifier) { | 
					
						
							|  |  |  |                     // the AST node under cursor is a specifier.
 | 
					
						
							|  |  |  |                     result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                         new SplitSimpleDeclarationOp(interface, index, declaration))); | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (core_declarator && interface->isCursorOn(core_declarator)) { | 
					
						
							|  |  |  |                     // got a core-declarator under the text cursor.
 | 
					
						
							|  |  |  |                     result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                         new SplitSimpleDeclarationOp(interface, index, declaration))); | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class AddBracesToIfOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     AddBracesToIfOp(const CppQuickFixInterface &interface, int priority, StatementAST *statement) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , _statement(statement) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Add Curly Braces")); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int start = currentFile->endOf(_statement->firstToken() - 1); | 
					
						
							|  |  |  |         changes.insert(start, QLatin1String(" {")); | 
					
						
							| 
									
										
										
										
											2011-01-07 14:49:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int end = currentFile->endOf(_statement->lastToken() - 1); | 
					
						
							|  |  |  |         changes.insert(end, QLatin1String("\n}")); | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(Utils::ChangeSet::Range(start, end)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     StatementAST *_statement; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AddBracesToIf::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // show when we're on the 'if' of an if statement
 | 
					
						
							|  |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     IfStatementAST *ifStatement = path.at(index)->asIfStatement(); | 
					
						
							|  |  |  |     if (ifStatement && interface->isCursorOn(ifStatement->if_token) && ifStatement->statement | 
					
						
							|  |  |  |         && ! ifStatement->statement->asCompoundStatement()) { | 
					
						
							|  |  |  |         result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |             new AddBracesToIfOp(interface, index, ifStatement->statement))); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // or if we're on the statement contained in the if
 | 
					
						
							|  |  |  |     // ### This may not be such a good idea, consider nested ifs...
 | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         IfStatementAST *ifStatement = path.at(index)->asIfStatement(); | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (ifStatement && ifStatement->statement | 
					
						
							|  |  |  |             && interface->isCursorOn(ifStatement->statement) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             && ! ifStatement->statement->asCompoundStatement()) { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new AddBracesToIfOp(interface, index, ifStatement->statement))); | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // ### This could very well be extended to the else branch
 | 
					
						
							|  |  |  |     // and other nodes entirely.
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class MoveDeclarationOutOfIfOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     MoveDeclarationOutOfIfOp(const CppQuickFixInterface &interface) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Move Declaration out of Condition")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         condition = mk.Condition(); | 
					
						
							|  |  |  |         pattern = mk.IfStatement(condition); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         changes.copy(currentFile->range(core), currentFile->startOf(condition)); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         int insertPos = currentFile->startOf(pattern); | 
					
						
							|  |  |  |         changes.move(currentFile->range(condition), insertPos); | 
					
						
							|  |  |  |         changes.insert(insertPos, QLatin1String(";\n")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(pattern)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ASTMatcher matcher; | 
					
						
							|  |  |  |     ASTPatternBuilder mk; | 
					
						
							|  |  |  |     CPPEditorWidget *editor; | 
					
						
							|  |  |  |     ConditionAST *condition; | 
					
						
							|  |  |  |     IfStatementAST *pattern; | 
					
						
							|  |  |  |     CoreDeclaratorAST *core; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void MoveDeclarationOutOfIf::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                    QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     typedef QSharedPointer<MoveDeclarationOutOfIfOp> Ptr; | 
					
						
							|  |  |  |     Ptr op(new MoveDeclarationOutOfIfOp(interface)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							|  |  |  |         if (IfStatementAST *statement = path.at(index)->asIfStatement()) { | 
					
						
							|  |  |  |             if (statement->match(op->pattern, &op->matcher) && op->condition->declarator) { | 
					
						
							|  |  |  |                 DeclaratorAST *declarator = op->condition->declarator; | 
					
						
							|  |  |  |                 op->core = declarator->core_declarator; | 
					
						
							|  |  |  |                 if (! op->core) | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (interface->isCursorOn(op->core)) { | 
					
						
							|  |  |  |                     op->setPriority(index); | 
					
						
							|  |  |  |                     result.append(op); | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class MoveDeclarationOutOfWhileOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     MoveDeclarationOutOfWhileOp(const CppQuickFixInterface &interface) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Move Declaration out of Condition")); | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         condition = mk.Condition(); | 
					
						
							|  |  |  |         pattern = mk.WhileStatement(condition); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         changes.insert(currentFile->startOf(condition), QLatin1String("(")); | 
					
						
							|  |  |  |         changes.insert(currentFile->endOf(condition), QLatin1String(") != 0")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         int insertPos = currentFile->startOf(pattern); | 
					
						
							|  |  |  |         const int conditionStart = currentFile->startOf(condition); | 
					
						
							|  |  |  |         changes.move(conditionStart, currentFile->startOf(core), insertPos); | 
					
						
							|  |  |  |         changes.copy(currentFile->range(core), insertPos); | 
					
						
							|  |  |  |         changes.insert(insertPos, QLatin1String(";\n")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(pattern)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ASTMatcher matcher; | 
					
						
							|  |  |  |     ASTPatternBuilder mk; | 
					
						
							|  |  |  |     CPPEditorWidget *editor; | 
					
						
							|  |  |  |     ConditionAST *condition; | 
					
						
							|  |  |  |     WhileStatementAST *pattern; | 
					
						
							|  |  |  |     CoreDeclaratorAST *core; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void MoveDeclarationOutOfWhile::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                       QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     QSharedPointer<MoveDeclarationOutOfWhileOp> op(new MoveDeclarationOutOfWhileOp(interface)); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							|  |  |  |         if (WhileStatementAST *statement = path.at(index)->asWhileStatement()) { | 
					
						
							|  |  |  |             if (statement->match(op->pattern, &op->matcher) && op->condition->declarator) { | 
					
						
							|  |  |  |                 DeclaratorAST *declarator = op->condition->declarator; | 
					
						
							|  |  |  |                 op->core = declarator->core_declarator; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (! op->core) | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (! declarator->equal_token) | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (! declarator->initializer) | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 if (interface->isCursorOn(op->core)) { | 
					
						
							|  |  |  |                     op->setPriority(index); | 
					
						
							|  |  |  |                     result.append(op); | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class SplitIfStatementOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     SplitIfStatementOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                        IfStatementAST *pattern, BinaryExpressionAST *condition) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , pattern(pattern) | 
					
						
							|  |  |  |         , condition(condition) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Split if Statement")); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const Token binaryToken = currentFile->tokenAt(condition->binary_op_token); | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (binaryToken.is(T_AMPER_AMPER)) | 
					
						
							|  |  |  |             splitAndCondition(currentFile); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             splitOrCondition(currentFile); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void splitAndCondition(CppRefactoringFilePtr currentFile) const | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         int startPos = currentFile->startOf(pattern); | 
					
						
							|  |  |  |         changes.insert(startPos, QLatin1String("if (")); | 
					
						
							|  |  |  |         changes.move(currentFile->range(condition->left_expression), startPos); | 
					
						
							|  |  |  |         changes.insert(startPos, QLatin1String(") {\n")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int lExprEnd = currentFile->endOf(condition->left_expression); | 
					
						
							|  |  |  |         changes.remove(lExprEnd, currentFile->startOf(condition->right_expression)); | 
					
						
							|  |  |  |         changes.insert(currentFile->endOf(pattern), QLatin1String("\n}")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(pattern)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void splitOrCondition(CppRefactoringFilePtr currentFile) const | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         StatementAST *ifTrueStatement = pattern->statement; | 
					
						
							|  |  |  |         CompoundStatementAST *compoundStatement = ifTrueStatement->asCompoundStatement(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         int insertPos = currentFile->endOf(ifTrueStatement); | 
					
						
							|  |  |  |         if (compoundStatement) | 
					
						
							|  |  |  |             changes.insert(insertPos, QLatin1String(" ")); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             changes.insert(insertPos, QLatin1String("\n")); | 
					
						
							|  |  |  |         changes.insert(insertPos, QLatin1String("else if (")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int rExprStart = currentFile->startOf(condition->right_expression); | 
					
						
							|  |  |  |         changes.move(rExprStart, currentFile->startOf(pattern->rparen_token), insertPos); | 
					
						
							|  |  |  |         changes.insert(insertPos, QLatin1String(")")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int rParenEnd = currentFile->endOf(pattern->rparen_token); | 
					
						
							|  |  |  |         changes.copy(rParenEnd, currentFile->endOf(pattern->statement), insertPos); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         const int lExprEnd = currentFile->endOf(condition->left_expression); | 
					
						
							|  |  |  |         changes.remove(lExprEnd, currentFile->startOf(condition->right_expression)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->appendIndentRange(currentFile->range(pattern)); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     IfStatementAST *pattern; | 
					
						
							|  |  |  |     BinaryExpressionAST *condition; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void SplitIfStatement::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     IfStatementAST *pattern = 0; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							|  |  |  |         AST *node = path.at(index); | 
					
						
							|  |  |  |         if (IfStatementAST *stmt = node->asIfStatement()) { | 
					
						
							|  |  |  |             pattern = stmt; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! pattern || ! pattern->statement) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     unsigned splitKind = 0; | 
					
						
							|  |  |  |     for (++index; index < path.size(); ++index) { | 
					
						
							|  |  |  |         AST *node = path.at(index); | 
					
						
							|  |  |  |         BinaryExpressionAST *condition = node->asBinaryExpression(); | 
					
						
							|  |  |  |         if (! condition) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         Token binaryToken = interface->currentFile()->tokenAt(condition->binary_op_token); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         // only accept a chain of ||s or &&s - no mixing
 | 
					
						
							|  |  |  |         if (! splitKind) { | 
					
						
							|  |  |  |             splitKind = binaryToken.kind(); | 
					
						
							|  |  |  |             if (splitKind != T_AMPER_AMPER && splitKind != T_PIPE_PIPE) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             // we can't reliably split &&s in ifs with an else branch
 | 
					
						
							|  |  |  |             if (splitKind == T_AMPER_AMPER && pattern->else_statement) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |         } else if (splitKind != binaryToken.kind()) { | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (interface->isCursorOn(condition->binary_op_token)) { | 
					
						
							|  |  |  |             result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new SplitIfStatementOp(interface, index, pattern, condition))); | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-09-30 10:45:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline QString msgQtStringLiteralDescription(const QString &replacement, int qtVersion) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return QApplication::translate("CppTools::QuickFix", "Enclose in %1(...) (Qt %2)") | 
					
						
							|  |  |  |            .arg(replacement).arg(qtVersion); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | static inline QString msgQtStringLiteralDescription(const QString &replacement) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return QApplication::translate("CppTools::QuickFix", "Enclose in %1(...)").arg(replacement); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Analze a string/character literal like "x", QLatin1String("x") and return the literal
 | 
					
						
							|  |  |  |  * (StringLiteral or NumericLiteral for characters) and its type | 
					
						
							|  |  |  |  * and the enclosing function (QLatin1String, tr...) */ | 
					
						
							|  |  |  | ExpressionAST *WrapStringLiteral::analyze(const QList<AST *> &path, | 
					
						
							|  |  |  |                                           const CppRefactoringFilePtr &file, | 
					
						
							|  |  |  |                                           Type *type, | 
					
						
							|  |  |  |                                           QByteArray *enclosingFunction /* = 0 */, | 
					
						
							|  |  |  |                                           CallAST **enclosingFunctionCall /* = 0 */) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     *type = TypeNone; | 
					
						
							|  |  |  |     if (enclosingFunction) | 
					
						
							|  |  |  |         enclosingFunction->clear(); | 
					
						
							|  |  |  |     if (enclosingFunctionCall) | 
					
						
							|  |  |  |         *enclosingFunctionCall = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (path.isEmpty()) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ExpressionAST *literal = path.last()->asExpression(); | 
					
						
							|  |  |  |     if (literal) { | 
					
						
							|  |  |  |         if (literal->asStringLiteral()) { | 
					
						
							|  |  |  |             // Check for Objective C string (@"bla")
 | 
					
						
							|  |  |  |             const QChar firstChar = file->charAt(file->startOf(literal)); | 
					
						
							|  |  |  |             *type = firstChar == QLatin1Char('@') ? TypeObjCString : TypeString; | 
					
						
							|  |  |  |         } else if (NumericLiteralAST *numericLiteral = literal->asNumericLiteral()) { | 
					
						
							|  |  |  |             // character ('c') constants are numeric.
 | 
					
						
							|  |  |  |             if (file->tokenAt(numericLiteral->literal_token).is(T_CHAR_LITERAL)) | 
					
						
							|  |  |  |                 *type = TypeChar; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     if (*type != TypeNone && enclosingFunction && path.size() > 1) { | 
					
						
							|  |  |  |         if (CallAST *call = path.at(path.size() - 2)->asCall()) { | 
					
						
							|  |  |  |             if (call->base_expression) { | 
					
						
							|  |  |  |                 if (IdExpressionAST *idExpr = call->base_expression->asIdExpression()) { | 
					
						
							|  |  |  |                     if (SimpleNameAST *functionName = idExpr->name->asSimpleName()) { | 
					
						
							|  |  |  |                         *enclosingFunction = file->tokenAt(functionName->identifier_token).identifier->chars(); | 
					
						
							|  |  |  |                         if (enclosingFunctionCall) | 
					
						
							|  |  |  |                             *enclosingFunctionCall = call; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return literal; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | /// Operation performs the operations of type ActionFlags passed in as actions.
 | 
					
						
							|  |  |  | class WrapStringLiteralOp : public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     typedef WrapStringLiteral Factory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     WrapStringLiteralOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                         unsigned actions, const QString &description, ExpressionAST *literal, | 
					
						
							|  |  |  |                         const QString &translationContext = QString()) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority), m_actions(actions), m_literal(literal), | 
					
						
							|  |  |  |           m_translationContext(translationContext) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(description); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const int startPos = currentFile->startOf(m_literal); | 
					
						
							|  |  |  |         const int endPos = currentFile->endOf(m_literal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // kill leading '@'. No need to adapt endPos, that is done by ChangeSet
 | 
					
						
							|  |  |  |         if (m_actions & Factory::RemoveObjectiveCAction) | 
					
						
							|  |  |  |             changes.remove(startPos, startPos + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Fix quotes
 | 
					
						
							|  |  |  |         if (m_actions & (Factory::SingleQuoteAction | Factory::DoubleQuoteAction)) { | 
					
						
							|  |  |  |             const QString newQuote((m_actions & Factory::SingleQuoteAction) ? QLatin1Char('\'') : QLatin1Char('"')); | 
					
						
							|  |  |  |             changes.replace(startPos, startPos + 1, newQuote); | 
					
						
							|  |  |  |             changes.replace(endPos - 1, endPos, newQuote); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Convert single character strings into character constants
 | 
					
						
							|  |  |  |         if (m_actions & Factory::ConvertEscapeSequencesToCharAction) { | 
					
						
							|  |  |  |             StringLiteralAST *stringLiteral = m_literal->asStringLiteral(); | 
					
						
							|  |  |  |             QTC_ASSERT(stringLiteral, return ;); | 
					
						
							|  |  |  |             const QByteArray oldContents(currentFile->tokenAt(stringLiteral->literal_token).identifier->chars()); | 
					
						
							|  |  |  |             const QByteArray newContents = Factory::stringToCharEscapeSequences(oldContents); | 
					
						
							|  |  |  |             QTC_ASSERT(!newContents.isEmpty(), return ;); | 
					
						
							|  |  |  |             if (oldContents != newContents) | 
					
						
							|  |  |  |                 changes.replace(startPos + 1, endPos -1, QString::fromLatin1(newContents)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Convert character constants into strings constants
 | 
					
						
							|  |  |  |         if (m_actions & Factory::ConvertEscapeSequencesToStringAction) { | 
					
						
							|  |  |  |             NumericLiteralAST *charLiteral = m_literal->asNumericLiteral(); // char 'c' constants are numerical.
 | 
					
						
							|  |  |  |             QTC_ASSERT(charLiteral, return ;); | 
					
						
							|  |  |  |             const QByteArray oldContents(currentFile->tokenAt(charLiteral->literal_token).identifier->chars()); | 
					
						
							|  |  |  |             const QByteArray newContents = Factory::charToStringEscapeSequences(oldContents); | 
					
						
							|  |  |  |             QTC_ASSERT(!newContents.isEmpty(), return ;); | 
					
						
							|  |  |  |             if (oldContents != newContents) | 
					
						
							|  |  |  |                 changes.replace(startPos + 1, endPos -1, QString::fromLatin1(newContents)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Enclose in literal or translation function, macro.
 | 
					
						
							|  |  |  |         if (m_actions & (Factory::EncloseActionMask | Factory::TranslationMask)) { | 
					
						
							|  |  |  |             changes.insert(endPos, QString(QLatin1Char(')'))); | 
					
						
							|  |  |  |             QString leading = Factory::replacement(m_actions); | 
					
						
							|  |  |  |             leading += QLatin1Char('('); | 
					
						
							|  |  |  |             if (m_actions & (Factory::TranslateQCoreApplicationAction | Factory::TranslateNoopAction)) { | 
					
						
							|  |  |  |                 leading += QLatin1Char('"'); | 
					
						
							|  |  |  |                 leading += m_translationContext; | 
					
						
							|  |  |  |                 leading += QLatin1String("\", "); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             changes.insert(startPos, leading); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     const unsigned m_actions; | 
					
						
							|  |  |  |     ExpressionAST *m_literal; | 
					
						
							|  |  |  |     const QString m_translationContext; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | void WrapStringLiteral::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     typedef CppQuickFixOperation::Ptr OperationPtr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Type type = TypeNone; | 
					
						
							|  |  |  |     QByteArray enclosingFunction; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ExpressionAST *literal = analyze(path, file, &type, &enclosingFunction); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     if (!literal || type == TypeNone) | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     if ((type == TypeChar && enclosingFunction == "QLatin1Char") | 
					
						
							|  |  |  |         || isQtStringLiteral(enclosingFunction) | 
					
						
							|  |  |  |         || isQtStringTranslation(enclosingFunction)) | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-09-30 10:45:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     const int priority = path.size() - 1; // very high priority
 | 
					
						
							|  |  |  |     if (type == TypeChar) { | 
					
						
							|  |  |  |         unsigned actions = EncloseInQLatin1CharAction; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         QString description = msgQtStringLiteralDescription(replacement(actions)); | 
					
						
							|  |  |  |         result << OperationPtr(new WrapStringLiteralOp(interface, priority, actions, | 
					
						
							|  |  |  |                                                              description, literal)); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |         if (NumericLiteralAST *charLiteral = literal->asNumericLiteral()) { | 
					
						
							|  |  |  |             const QByteArray contents(file->tokenAt(charLiteral->literal_token).identifier->chars()); | 
					
						
							|  |  |  |             if (!charToStringEscapeSequences(contents).isEmpty()) { | 
					
						
							|  |  |  |                 actions = DoubleQuoteAction | ConvertEscapeSequencesToStringAction; | 
					
						
							|  |  |  |                 description = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                               "Convert to String Literal"); | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 result << OperationPtr(new WrapStringLiteralOp(interface, priority, actions, | 
					
						
							|  |  |  |                                                                      description, literal)); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-09-30 10:45:11 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         const unsigned objectiveCActions = type == TypeObjCString ? | 
					
						
							|  |  |  |                                            unsigned(RemoveObjectiveCAction) : 0u; | 
					
						
							|  |  |  |         unsigned actions = 0; | 
					
						
							|  |  |  |         if (StringLiteralAST *stringLiteral = literal->asStringLiteral()) { | 
					
						
							|  |  |  |             const QByteArray contents(file->tokenAt(stringLiteral->literal_token).identifier->chars()); | 
					
						
							|  |  |  |             if (!stringToCharEscapeSequences(contents).isEmpty()) { | 
					
						
							|  |  |  |                 actions = EncloseInQLatin1CharAction | SingleQuoteAction | 
					
						
							|  |  |  |                           | ConvertEscapeSequencesToCharAction | objectiveCActions; | 
					
						
							|  |  |  |                 QString description = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                       "Convert to Character Literal and Enclose in QLatin1Char(...)"); | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 result << OperationPtr(new WrapStringLiteralOp(interface, priority, | 
					
						
							|  |  |  |                                                                      actions, description, literal)); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |                 actions &= ~EncloseInQLatin1CharAction; | 
					
						
							|  |  |  |                 description = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                               "Convert to Character Literal"); | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 result << OperationPtr(new WrapStringLiteralOp(interface, priority, | 
					
						
							|  |  |  |                                                                      actions, description, literal)); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         actions = EncloseInQLatin1StringAction | objectiveCActions; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         result << OperationPtr( | 
					
						
							|  |  |  |             new WrapStringLiteralOp(interface, priority, actions, | 
					
						
							|  |  |  |                 msgQtStringLiteralDescription(replacement(actions), 4), | 
					
						
							|  |  |  |                     literal)); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |         actions = EncloseInQStringLiteralAction | objectiveCActions; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         result << OperationPtr( | 
					
						
							|  |  |  |             new WrapStringLiteralOp(interface, priority, actions, | 
					
						
							|  |  |  |                 msgQtStringLiteralDescription(replacement(actions), 5), literal)); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | QString WrapStringLiteral::replacement(unsigned actions) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (actions & EncloseInQLatin1CharAction) | 
					
						
							|  |  |  |         return QLatin1String("QLatin1Char"); | 
					
						
							|  |  |  |     if (actions & EncloseInQLatin1StringAction) | 
					
						
							|  |  |  |         return QLatin1String("QLatin1String"); | 
					
						
							|  |  |  |     if (actions & EncloseInQStringLiteralAction) | 
					
						
							|  |  |  |         return QLatin1String("QStringLiteral"); | 
					
						
							|  |  |  |     if (actions & TranslateTrAction) | 
					
						
							|  |  |  |         return QLatin1String("tr"); | 
					
						
							|  |  |  |     if (actions & TranslateQCoreApplicationAction) | 
					
						
							|  |  |  |         return QLatin1String("QCoreApplication::translate"); | 
					
						
							|  |  |  |     if (actions & TranslateNoopAction) | 
					
						
							|  |  |  |         return QLatin1String("QT_TRANSLATE_NOOP"); | 
					
						
							|  |  |  |     return QString(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | /* Convert single-character string literals into character literals with some
 | 
					
						
							|  |  |  |  * special cases "a" --> 'a', "'" --> '\'', "\n" --> '\n', "\"" --> '"'. */ | 
					
						
							|  |  |  | QByteArray WrapStringLiteral::stringToCharEscapeSequences(const QByteArray &content) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (content.size() == 1) | 
					
						
							|  |  |  |         return content.at(0) == '\'' ? QByteArray("\\'") : content; | 
					
						
							|  |  |  |     if (content.size() == 2 && content.at(0) == '\\') | 
					
						
							|  |  |  |         return content == "\\\"" ? QByteArray(1, '"') : content; | 
					
						
							|  |  |  |     return QByteArray(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | /* Convert character literal into a string literal with some special cases
 | 
					
						
							|  |  |  |  * 'a' -> "a", '\n' -> "\n", '\'' --> "'", '"' --> "\"". */ | 
					
						
							|  |  |  | QByteArray WrapStringLiteral::charToStringEscapeSequences(const QByteArray &content) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (content.size() == 1) | 
					
						
							|  |  |  |         return content.at(0) == '"' ? QByteArray("\\\"") : content; | 
					
						
							|  |  |  |     if (content.size() == 2) | 
					
						
							|  |  |  |         return content == "\\'" ? QByteArray("'") : content; | 
					
						
							|  |  |  |     return QByteArray(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void TranslateStringLiteral::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                    QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // Initialize
 | 
					
						
							|  |  |  |     WrapStringLiteral::Type type = WrapStringLiteral::TypeNone; | 
					
						
							|  |  |  |     QByteArray enclosingFunction; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							|  |  |  |     ExpressionAST *literal = WrapStringLiteral::analyze(path, file, &type, &enclosingFunction); | 
					
						
							|  |  |  |     if (!literal || type != WrapStringLiteral::TypeString | 
					
						
							|  |  |  |        || isQtStringLiteral(enclosingFunction) || isQtStringTranslation(enclosingFunction)) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString trContext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QSharedPointer<Control> control = interface->context().control(); | 
					
						
							|  |  |  |     const Name *trName = control->identifier("tr"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Check whether we are in a method:
 | 
					
						
							|  |  |  |     const QString description = QApplication::translate("CppTools::QuickFix", "Mark as Translatable"); | 
					
						
							|  |  |  |     for (int i = path.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         if (FunctionDefinitionAST *definition = path.at(i)->asFunctionDefinition()) { | 
					
						
							|  |  |  |             Function *function = definition->symbol; | 
					
						
							|  |  |  |             ClassOrNamespace *b = interface->context().lookupType(function); | 
					
						
							|  |  |  |             if (b) { | 
					
						
							|  |  |  |                 // Do we have a tr method?
 | 
					
						
							|  |  |  |                 foreach (const LookupItem &r, b->find(trName)) { | 
					
						
							|  |  |  |                     Symbol *s = r.declaration(); | 
					
						
							|  |  |  |                     if (s->type()->isFunctionType()) { | 
					
						
							|  |  |  |                         // no context required for tr
 | 
					
						
							|  |  |  |                         result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                             new WrapStringLiteralOp(interface, path.size() - 1, | 
					
						
							|  |  |  |                                                           WrapStringLiteral::TranslateTrAction, | 
					
						
							|  |  |  |                                                           description, literal))); | 
					
						
							|  |  |  |                         return; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             // We need to do a QCA::translate, so we need a context.
 | 
					
						
							|  |  |  |             // Use fully qualified class name:
 | 
					
						
							|  |  |  |             Overview oo; | 
					
						
							|  |  |  |             foreach (const Name *n, LookupContext::path(function)) { | 
					
						
							|  |  |  |                 if (!trContext.isEmpty()) | 
					
						
							|  |  |  |                     trContext.append(QLatin1String("::")); | 
					
						
							|  |  |  |                 trContext.append(oo.prettyName(n)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             // ... or global if none available!
 | 
					
						
							|  |  |  |             if (trContext.isEmpty()) | 
					
						
							|  |  |  |                 trContext = QLatin1String("GLOBAL"); | 
					
						
							|  |  |  |             result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new WrapStringLiteralOp(interface, path.size() - 1, | 
					
						
							|  |  |  |                                               WrapStringLiteral::TranslateQCoreApplicationAction, | 
					
						
							|  |  |  |                                               description, literal, trContext))); | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // We need to use Q_TRANSLATE_NOOP
 | 
					
						
							|  |  |  |     result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |         new WrapStringLiteralOp(interface, path.size() - 1, | 
					
						
							|  |  |  |                                       WrapStringLiteral::TranslateNoopAction, | 
					
						
							|  |  |  |                                       description, literal, trContext))); | 
					
						
							| 
									
										
										
										
											2011-12-14 14:23:25 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class ConvertCStringToNSStringOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ConvertCStringToNSStringOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                                StringLiteralAST *stringLiteral, CallAST *qlatin1Call) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , stringLiteral(stringLiteral) | 
					
						
							|  |  |  |         , qlatin1Call(qlatin1Call) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Convert to Objective-C String Literal")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-08-13 12:49:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (qlatin1Call) { | 
					
						
							|  |  |  |             changes.replace(currentFile->startOf(qlatin1Call), currentFile->startOf(stringLiteral), QLatin1String("@")); | 
					
						
							|  |  |  |             changes.remove(currentFile->endOf(stringLiteral), currentFile->endOf(qlatin1Call)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             changes.insert(currentFile->startOf(stringLiteral), QLatin1String("@")); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     StringLiteralAST *stringLiteral; | 
					
						
							|  |  |  |     CallAST *qlatin1Call; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void ConvertCStringToNSString::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                      QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (interface->editor()->mimeType() != QLatin1String(CppTools::Constants::OBJECTIVE_CPP_SOURCE_MIMETYPE)) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     WrapStringLiteral::Type type = WrapStringLiteral::TypeNone; | 
					
						
							|  |  |  |     QByteArray enclosingFunction; | 
					
						
							|  |  |  |     CallAST *qlatin1Call; | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     ExpressionAST *literal = WrapStringLiteral::analyze(path, file, &type, &enclosingFunction, &qlatin1Call); | 
					
						
							|  |  |  |     if (!literal || type != WrapStringLiteral::TypeString) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (!isQtStringLiteral(enclosingFunction)) | 
					
						
							|  |  |  |         qlatin1Call = 0; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |         new ConvertCStringToNSStringOp(interface, path.size() - 1, literal->asStringLiteral(), | 
					
						
							|  |  |  |             qlatin1Call))); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class ConvertNumericLiteralOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ConvertNumericLiteralOp(const CppQuickFixInterface &interface, int start, int end, | 
					
						
							|  |  |  |                             const QString &replacement) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |         , start(start) | 
					
						
							|  |  |  |         , end(end) | 
					
						
							|  |  |  |         , replacement(replacement) | 
					
						
							|  |  |  |     {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         changes.replace(start, end, replacement); | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     int start, end; | 
					
						
							|  |  |  |     QString replacement; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void ConvertNumericLiteral::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (path.isEmpty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     NumericLiteralAST *literal = path.last()->asNumericLiteral(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (! literal) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     Token token = file->tokenAt(literal->asNumericLiteral()->literal_token); | 
					
						
							|  |  |  |     if (!token.is(T_NUMERIC_LITERAL)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     const NumericLiteral *numeric = token.number; | 
					
						
							|  |  |  |     if (numeric->isDouble() || numeric->isFloat()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // remove trailing L or U and stuff
 | 
					
						
							|  |  |  |     const char * const spell = numeric->chars(); | 
					
						
							|  |  |  |     int numberLength = numeric->size(); | 
					
						
							|  |  |  |     while (numberLength > 0 && !std::isxdigit(spell[numberLength - 1])) | 
					
						
							|  |  |  |         --numberLength; | 
					
						
							|  |  |  |     if (numberLength < 1) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // convert to number
 | 
					
						
							|  |  |  |     bool valid; | 
					
						
							|  |  |  |     ulong value = QString::fromUtf8(spell).left(numberLength).toULong(&valid, 0); | 
					
						
							|  |  |  |     if (!valid) // e.g. octal with digit > 7
 | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const int priority = path.size() - 1; // very high priority
 | 
					
						
							|  |  |  |     const int start = file->startOf(literal); | 
					
						
							|  |  |  |     const char * const str = numeric->chars(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!numeric->isHex()) { | 
					
						
							|  |  |  |         /*
 | 
					
						
							|  |  |  |           Convert integer literal to hex representation. | 
					
						
							|  |  |  |           Replace | 
					
						
							|  |  |  |             32 | 
					
						
							|  |  |  |             040 | 
					
						
							|  |  |  |           With | 
					
						
							|  |  |  |             0x20 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         */ | 
					
						
							|  |  |  |         QString replacement; | 
					
						
							|  |  |  |         replacement.sprintf("0x%lX", value); | 
					
						
							|  |  |  |         QuickFixOperation::Ptr op( | 
					
						
							|  |  |  |             new ConvertNumericLiteralOp(interface, start, start + numberLength, replacement)); | 
					
						
							|  |  |  |         op->setDescription(QApplication::translate("CppTools::QuickFix", "Convert to Hexadecimal")); | 
					
						
							|  |  |  |         op->setPriority(priority); | 
					
						
							|  |  |  |         result.append(op); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value != 0) { | 
					
						
							|  |  |  |         if (!(numberLength > 1 && str[0] == '0' && str[1] != 'x' && str[1] != 'X')) { | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             /*
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |               Convert integer literal to octal representation. | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |               Replace | 
					
						
							|  |  |  |                 32 | 
					
						
							|  |  |  |                 0x20 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |               With | 
					
						
							|  |  |  |                 040 | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             */ | 
					
						
							|  |  |  |             QString replacement; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             replacement.sprintf("0%lo", value); | 
					
						
							|  |  |  |             QuickFixOperation::Ptr op( | 
					
						
							|  |  |  |                 new ConvertNumericLiteralOp(interface, start, start + numberLength, replacement)); | 
					
						
							|  |  |  |             op->setDescription(QApplication::translate("CppTools::QuickFix", "Convert to Octal")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             op->setPriority(priority); | 
					
						
							|  |  |  |             result.append(op); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (value != 0 || numeric->isHex()) { | 
					
						
							|  |  |  |         if (!(numberLength > 1 && str[0] != '0')) { | 
					
						
							|  |  |  |             /*
 | 
					
						
							|  |  |  |               Convert integer literal to decimal representation. | 
					
						
							|  |  |  |               Replace | 
					
						
							|  |  |  |                 0x20 | 
					
						
							|  |  |  |                 040 | 
					
						
							|  |  |  |               With | 
					
						
							|  |  |  |                 32 | 
					
						
							|  |  |  |             */ | 
					
						
							|  |  |  |             QString replacement; | 
					
						
							|  |  |  |             replacement.sprintf("%lu", value); | 
					
						
							|  |  |  |             QuickFixOperation::Ptr op( | 
					
						
							|  |  |  |                 new ConvertNumericLiteralOp(interface, start, start + numberLength, replacement)); | 
					
						
							|  |  |  |             op->setDescription(QApplication::translate("CppTools::QuickFix", "Convert to Decimal")); | 
					
						
							|  |  |  |             op->setPriority(priority); | 
					
						
							|  |  |  |             result.append(op); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class AddIncludeForForwardDeclarationOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     AddIncludeForForwardDeclarationOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                                       Symbol *fwdClass) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , fwdClass(fwdClass) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "#include Header File")); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         QTC_ASSERT(fwdClass != 0, return); | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CppTools::SymbolFinder symbolFinder; | 
					
						
							|  |  |  |         if (Class *k = symbolFinder.findMatchingClassDeclaration(fwdClass, snapshot())) { | 
					
						
							|  |  |  |             const QString headerFile = QString::fromUtf8(k->fileName(), k->fileNameLength()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // collect the fwd headers
 | 
					
						
							|  |  |  |             Snapshot fwdHeaders; | 
					
						
							|  |  |  |             fwdHeaders.insert(snapshot().document(headerFile)); | 
					
						
							|  |  |  |             foreach (Document::Ptr doc, snapshot()) { | 
					
						
							|  |  |  |                 QFileInfo headerFileInfo(doc->fileName()); | 
					
						
							|  |  |  |                 if (doc->globalSymbolCount() == 0 && doc->includes().size() == 1) | 
					
						
							|  |  |  |                     fwdHeaders.insert(doc); | 
					
						
							|  |  |  |                 else if (headerFileInfo.suffix().isEmpty()) | 
					
						
							|  |  |  |                     fwdHeaders.insert(doc); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             DependencyTable dep; | 
					
						
							|  |  |  |             dep.build(fwdHeaders); | 
					
						
							|  |  |  |             QStringList candidates = dep.dependencyTable().value(headerFile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const QString className = QString::fromUtf8(k->identifier()->chars()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             QString best; | 
					
						
							|  |  |  |             foreach (const QString &c, candidates) { | 
					
						
							|  |  |  |                 QFileInfo headerFileInfo(c); | 
					
						
							|  |  |  |                 if (headerFileInfo.fileName() == className) { | 
					
						
							|  |  |  |                     best = c; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } else if (headerFileInfo.fileName().at(0).isUpper()) { | 
					
						
							|  |  |  |                     best = c; | 
					
						
							|  |  |  |                     // and continue
 | 
					
						
							|  |  |  |                 } else if (! best.isEmpty()) { | 
					
						
							|  |  |  |                     if (c.count(QLatin1Char('/')) < best.count(QLatin1Char('/'))) | 
					
						
							|  |  |  |                         best = c; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             if (best.isEmpty()) | 
					
						
							|  |  |  |                 best = headerFile; | 
					
						
							| 
									
										
										
										
											2011-01-07 16:55:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             int pos = currentFile->startOf(1); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             unsigned currentLine = currentFile->cursor().blockNumber() + 1; | 
					
						
							|  |  |  |             unsigned bestLine = 0; | 
					
						
							|  |  |  |             foreach (const Document::Include &incl, assistInterface()->semanticInfo().doc->includes()) { | 
					
						
							|  |  |  |                 if (incl.line() < currentLine) | 
					
						
							|  |  |  |                     bestLine = incl.line(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (bestLine) | 
					
						
							|  |  |  |                 pos = currentFile->document()->findBlockByNumber(bestLine).position(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Utils::ChangeSet changes; | 
					
						
							|  |  |  |             changes.insert(pos, QLatin1String("#include <") | 
					
						
							|  |  |  |                            + QFileInfo(best).fileName() + QLatin1String(">\n")); | 
					
						
							|  |  |  |             currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |             currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  |     static Symbol *checkName(const CppQuickFixInterface &interface, NameAST *ast) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-04-15 16:19:23 +02:00
										 |  |  |         if (ast && interface->isCursorOn(ast)) { | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             if (const Name *name = ast->name) { | 
					
						
							|  |  |  |                 unsigned line, column; | 
					
						
							| 
									
										
										
										
											2011-04-15 16:19:23 +02:00
										 |  |  |                 interface->semanticInfo().doc->translationUnit()->getTokenStartPosition(ast->firstToken(), &line, &column); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 Symbol *fwdClass = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-15 16:19:23 +02:00
										 |  |  |                 foreach (const LookupItem &r, | 
					
						
							|  |  |  |                          interface->context().lookup(name, | 
					
						
							|  |  |  |                                                      interface->semanticInfo().doc->scopeAt(line, column))) { | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                     if (! r.declaration()) | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     else if (ForwardClassDeclaration *fwd = r.declaration()->asForwardClassDeclaration()) | 
					
						
							|  |  |  |                         fwdClass = fwd; | 
					
						
							|  |  |  |                     else if (r.declaration()->isClass()) | 
					
						
							|  |  |  |                         return 0; // nothing to do.
 | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return fwdClass; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     Symbol *fwdClass; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void AddIncludeForForwardDeclaration::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                             QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     for (int index = path.size() - 1; index != -1; --index) { | 
					
						
							|  |  |  |         AST *ast = path.at(index); | 
					
						
							|  |  |  |         if (NamedTypeSpecifierAST *namedTy = ast->asNamedTypeSpecifier()) { | 
					
						
							|  |  |  |             if (Symbol *fwdClass = AddIncludeForForwardDeclarationOp::checkName(interface, namedTy->name)) { | 
					
						
							|  |  |  |                 result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                     new AddIncludeForForwardDeclarationOp(interface, index, fwdClass))); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if (ElaboratedTypeSpecifierAST *eTy = ast->asElaboratedTypeSpecifier()) { | 
					
						
							|  |  |  |             if (Symbol *fwdClass = AddIncludeForForwardDeclarationOp::checkName(interface, eTy->name)) { | 
					
						
							|  |  |  |                 result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                     new AddIncludeForForwardDeclarationOp(interface, index, fwdClass))); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class AddLocalDeclarationOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     AddLocalDeclarationOp(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                           int priority, | 
					
						
							|  |  |  |                           const BinaryExpressionAST *binaryAST, | 
					
						
							|  |  |  |                           const SimpleNameAST *simpleNameAST) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , binaryAST(binaryAST) | 
					
						
							|  |  |  |         , simpleNameAST(simpleNameAST) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", "Add Local Declaration")); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         TypeOfExpression typeOfExpression; | 
					
						
							|  |  |  |         typeOfExpression.init(assistInterface()->semanticInfo().doc, | 
					
						
							|  |  |  |                               snapshot(), assistInterface()->context().bindings()); | 
					
						
							|  |  |  |         Scope *scope = currentFile->scopeAt(binaryAST->firstToken()); | 
					
						
							|  |  |  |         const QList<LookupItem> result = | 
					
						
							|  |  |  |                 typeOfExpression(currentFile->textOf(binaryAST->right_expression).toUtf8(), | 
					
						
							|  |  |  |                                  scope, | 
					
						
							|  |  |  |                                  TypeOfExpression::Preprocess); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (! result.isEmpty()) { | 
					
						
							|  |  |  |             SubstitutionEnvironment env; | 
					
						
							|  |  |  |             env.setContext(assistInterface()->context()); | 
					
						
							|  |  |  |             env.switchScope(result.first().scope()); | 
					
						
							|  |  |  |             ClassOrNamespace *con = typeOfExpression.context().lookupType(scope); | 
					
						
							|  |  |  |             if (!con) | 
					
						
							|  |  |  |                 con = typeOfExpression.context().globalNamespace(); | 
					
						
							|  |  |  |             UseMinimalNames q(con); | 
					
						
							|  |  |  |             env.enter(&q); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Control *control = assistInterface()->context().control().data(); | 
					
						
							|  |  |  |             FullySpecifiedType tn = rewriteType(result.first().type(), &env, control); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Overview oo = CppCodeStyleSettings::currentProjectCodeStyleOverview(); | 
					
						
							|  |  |  |             QString ty = oo.prettyType(tn, simpleNameAST->name); | 
					
						
							|  |  |  |             if (! ty.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                 Utils::ChangeSet changes; | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 changes.replace(currentFile->startOf(binaryAST), | 
					
						
							|  |  |  |                                 currentFile->endOf(simpleNameAST), | 
					
						
							|  |  |  |                                 ty); | 
					
						
							| 
									
										
										
										
											2011-08-17 11:35:57 +02:00
										 |  |  |                 currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |                 currentFile->apply(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     const BinaryExpressionAST *binaryAST; | 
					
						
							|  |  |  |     const SimpleNameAST *simpleNameAST; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void AddLocalDeclaration::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int index = path.size() - 1; index != -1; --index) { | 
					
						
							|  |  |  |         if (BinaryExpressionAST *binary = path.at(index)->asBinaryExpression()) { | 
					
						
							|  |  |  |             if (binary->left_expression && binary->right_expression && file->tokenAt(binary->binary_op_token).is(T_EQUAL)) { | 
					
						
							|  |  |  |                 IdExpressionAST *idExpr = binary->left_expression->asIdExpression(); | 
					
						
							|  |  |  |                 if (interface->isCursorOn(binary->left_expression) && idExpr && idExpr->name->asSimpleName() != 0) { | 
					
						
							|  |  |  |                     SimpleNameAST *nameAST = idExpr->name->asSimpleName(); | 
					
						
							|  |  |  |                     const QList<LookupItem> results = interface->context().lookup(nameAST->name, file->scopeAt(nameAST->firstToken())); | 
					
						
							|  |  |  |                     Declaration *decl = 0; | 
					
						
							|  |  |  |                     foreach (const LookupItem &r, results) { | 
					
						
							|  |  |  |                         if (! r.declaration()) | 
					
						
							|  |  |  |                             continue; | 
					
						
							|  |  |  |                         else if (Declaration *d = r.declaration()->asDeclaration()) { | 
					
						
							|  |  |  |                             if (! d->type()->isFunctionType()) { | 
					
						
							|  |  |  |                                 decl = d; | 
					
						
							|  |  |  |                                 break; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                             } | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                     if (! decl) { | 
					
						
							|  |  |  |                         result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                             new AddLocalDeclarationOp(interface, index, binary, nameAST))); | 
					
						
							|  |  |  |                         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class ConvertToCamelCaseOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ConvertToCamelCaseOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                          const QString &newName) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , m_name(newName) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Convert to Camel Case")); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int i = 1; i < m_name.length(); ++i) { | 
					
						
							|  |  |  |             QCharRef c = m_name[i]; | 
					
						
							|  |  |  |             if (c.isUpper()) { | 
					
						
							|  |  |  |                 c = c.toLower(); | 
					
						
							|  |  |  |             } else if (i < m_name.length() - 1 | 
					
						
							|  |  |  |                        && isConvertibleUnderscore(m_name, i)) { | 
					
						
							|  |  |  |                 m_name.remove(i, 1); | 
					
						
							|  |  |  |                 m_name[i] = m_name.at(i).toUpper(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         static_cast<CppEditor::Internal::CPPEditorWidget*>(assistInterface()->editor())->renameUsagesNow(m_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static bool isConvertibleUnderscore(const QString &name, int pos) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return name.at(pos) == QLatin1Char('_') && name.at(pos+1).isLetter() | 
					
						
							|  |  |  |                 && !(pos == 1 && name.at(0) == QLatin1Char('m')); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     QString m_name; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void ConvertToCamelCase::match(const CppQuickFixInterface &interface, QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (path.isEmpty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     AST * const ast = path.last(); | 
					
						
							|  |  |  |     const Name *name = 0; | 
					
						
							|  |  |  |     if (const NameAST * const nameAst = ast->asName()) { | 
					
						
							|  |  |  |         if (nameAst->name && nameAst->name->asNameId()) | 
					
						
							|  |  |  |             name = nameAst->name; | 
					
						
							|  |  |  |     } else if (const NamespaceAST * const namespaceAst = ast->asNamespace()) { | 
					
						
							|  |  |  |         name = namespaceAst->symbol->name(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (!name) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     QString newName = QString::fromUtf8(name->identifier()->chars()); | 
					
						
							|  |  |  |     if (newName.length() < 3) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     for (int i = 1; i < newName.length() - 1; ++i) { | 
					
						
							|  |  |  |         if (ConvertToCamelCaseOp::isConvertibleUnderscore(newName, i)) { | 
					
						
							|  |  |  |             result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new ConvertToCamelCaseOp(interface, path.size() - 1, newName))); | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class AddIncludeForUndefinedIdentifierOp: public CppQuickFixOperation | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     AddIncludeForUndefinedIdentifierOp(const CppQuickFixInterface &interface, int priority, | 
					
						
							|  |  |  |                                        const QString &include) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface, priority) | 
					
						
							|  |  |  |         , m_include(include) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         setDescription(QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                "Add #include %1").arg(m_include)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr file = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // find location of last include in file
 | 
					
						
							|  |  |  |         QList<Document::Include> includes = file->cppDocument()->includes(); | 
					
						
							|  |  |  |         unsigned lastIncludeLine = 0; | 
					
						
							|  |  |  |         foreach (const Document::Include &include, includes) { | 
					
						
							|  |  |  |             if (include.line() > lastIncludeLine) | 
					
						
							|  |  |  |                 lastIncludeLine = include.line(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // add include
 | 
					
						
							|  |  |  |         const int insertPos = file->position(lastIncludeLine + 1, 1) - 1; | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         changes.insert(insertPos, QLatin1String("\n#include ") + m_include); | 
					
						
							|  |  |  |         file->setChangeSet(changes); | 
					
						
							|  |  |  |         file->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     QString m_include; | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void AddIncludeForUndefinedIdentifier::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                              QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     CppClassesFilter *classesFilter = ExtensionSystem::PluginManager::getObject<CppClassesFilter>(); | 
					
						
							|  |  |  |     if (!classesFilter) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (path.isEmpty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // find the largest enclosing Name
 | 
					
						
							|  |  |  |     const NameAST *enclosingName = 0; | 
					
						
							|  |  |  |     const SimpleNameAST *innermostName = 0; | 
					
						
							|  |  |  |     for (int i = path.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         if (NameAST *nameAst = path.at(i)->asName()) { | 
					
						
							|  |  |  |             enclosingName = nameAst; | 
					
						
							|  |  |  |             if (!innermostName) { | 
					
						
							|  |  |  |                 innermostName = nameAst->asSimpleName(); | 
					
						
							|  |  |  |                 if (!innermostName) | 
					
						
							|  |  |  |                     return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (!enclosingName || !enclosingName->name) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // find the enclosing scope
 | 
					
						
							|  |  |  |     unsigned line, column; | 
					
						
							|  |  |  |     const Document::Ptr &doc = interface->semanticInfo().doc; | 
					
						
							|  |  |  |     doc->translationUnit()->getTokenStartPosition(enclosingName->firstToken(), &line, &column); | 
					
						
							|  |  |  |     Scope *scope = doc->scopeAt(line, column); | 
					
						
							|  |  |  |     if (!scope) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // check if the name resolves to something
 | 
					
						
							|  |  |  |     QList<LookupItem> existingResults = interface->context().lookup(enclosingName->name, scope); | 
					
						
							|  |  |  |     if (!existingResults.isEmpty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QString &className = Overview().prettyName(innermostName->name); | 
					
						
							|  |  |  |     if (className.isEmpty()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // find the include paths
 | 
					
						
							|  |  |  |     QStringList includePaths; | 
					
						
							|  |  |  |     CppModelManagerInterface *modelManager = CppModelManagerInterface::instance(); | 
					
						
							|  |  |  |     QList<CppModelManagerInterface::ProjectInfo> projectInfos = modelManager->projectInfos(); | 
					
						
							|  |  |  |     bool inProject = false; | 
					
						
							|  |  |  |     foreach (const CppModelManagerInterface::ProjectInfo &info, projectInfos) { | 
					
						
							| 
									
										
										
										
											2013-03-04 01:30:46 +04:00
										 |  |  |         foreach (ProjectPart::Ptr part, info.projectParts()) { | 
					
						
							|  |  |  |             foreach (const ProjectFile &file, part->files) { | 
					
						
							|  |  |  |                 if (file.path == doc->fileName()) { | 
					
						
							|  |  |  |                     inProject = true; | 
					
						
							|  |  |  |                     includePaths += part->includePaths; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (!inProject) { | 
					
						
							|  |  |  |         // better use all include paths than none
 | 
					
						
							|  |  |  |         foreach (const CppModelManagerInterface::ProjectInfo &info, projectInfos) { | 
					
						
							| 
									
										
										
										
											2013-03-04 01:30:46 +04:00
										 |  |  |             foreach (ProjectPart::Ptr part, info.projectParts()) | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |                 includePaths += part->includePaths; | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // find a include file through the locator
 | 
					
						
							|  |  |  |     QFutureInterface<Locator::FilterEntry> dummyInterface; | 
					
						
							|  |  |  |     QList<Locator::FilterEntry> matches = classesFilter->matchesFor(dummyInterface, className); | 
					
						
							|  |  |  |     bool classExists = false; | 
					
						
							|  |  |  |     foreach (const Locator::FilterEntry &entry, matches) { | 
					
						
							|  |  |  |         const ModelItemInfo info = entry.internalData.value<ModelItemInfo>(); | 
					
						
							|  |  |  |         if (info.symbolName != className) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         classExists = true; | 
					
						
							|  |  |  |         const QString &fileName = info.fileName; | 
					
						
							|  |  |  |         const QFileInfo fileInfo(fileName); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // find the shortest way to include fileName given the includePaths
 | 
					
						
							|  |  |  |         QString shortestInclude; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (fileInfo.path() == QFileInfo(doc->fileName()).path()) { | 
					
						
							|  |  |  |             shortestInclude = QLatin1Char('"') + fileInfo.fileName() + QLatin1Char('"'); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             foreach (const QString &includePath, includePaths) { | 
					
						
							|  |  |  |                 if (!fileName.startsWith(includePath)) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 QString relativePath = fileName.mid(includePath.size()); | 
					
						
							|  |  |  |                 if (!relativePath.isEmpty() && relativePath.at(0) == QLatin1Char('/')) | 
					
						
							|  |  |  |                     relativePath = relativePath.mid(1); | 
					
						
							|  |  |  |                 if (shortestInclude.isEmpty() || relativePath.size() + 2 < shortestInclude.size()) | 
					
						
							|  |  |  |                     shortestInclude = QLatin1Char('<') + relativePath + QLatin1Char('>'); | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         if (!shortestInclude.isEmpty()) { | 
					
						
							|  |  |  |             result += CppQuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new AddIncludeForUndefinedIdentifierOp(interface, 0, shortestInclude)); | 
					
						
							| 
									
										
										
										
											2011-11-25 12:05:58 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     // for QSomething, propose a <QSomething> include -- if such a class was in the locator
 | 
					
						
							|  |  |  |     if (classExists | 
					
						
							|  |  |  |             && className.size() > 2 | 
					
						
							|  |  |  |             && className.at(0) == QLatin1Char('Q') | 
					
						
							|  |  |  |             && className.at(1).isUpper()) { | 
					
						
							|  |  |  |         const QString include = QLatin1Char('<') + className + QLatin1Char('>'); | 
					
						
							|  |  |  |         result += CppQuickFixOperation::Ptr( | 
					
						
							|  |  |  |             new AddIncludeForUndefinedIdentifierOp(interface, 1, include)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class RearrangeParamDeclarationListOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     enum Target { TargetPrevious, TargetNext }; | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     RearrangeParamDeclarationListOp(const CppQuickFixInterface &interface, AST *currentParam, | 
					
						
							|  |  |  |                                     AST *targetParam, Target target) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |         , m_currentParam(currentParam) | 
					
						
							|  |  |  |         , m_targetParam(targetParam) | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         QString targetString; | 
					
						
							|  |  |  |         if (target == TargetPrevious) | 
					
						
							|  |  |  |             targetString = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                    "Switch with Previous Parameter"); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             targetString = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                                                    "Switch with Next Parameter"); | 
					
						
							|  |  |  |         setDescription(targetString); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         int targetEndPos = currentFile->endOf(m_targetParam); | 
					
						
							|  |  |  |         ChangeSet changes; | 
					
						
							|  |  |  |         changes.flip(currentFile->startOf(m_currentParam), currentFile->endOf(m_currentParam), | 
					
						
							|  |  |  |                      currentFile->startOf(m_targetParam), targetEndPos); | 
					
						
							|  |  |  |         currentFile->setChangeSet(changes); | 
					
						
							|  |  |  |         currentFile->setOpenEditor(false, targetEndPos); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     AST *m_currentParam; | 
					
						
							|  |  |  |     AST *m_targetParam; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void RearrangeParamDeclarationList::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                           QuickFixOperations &result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const QList<AST *> path = interface->path(); | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ParameterDeclarationAST *paramDecl = 0; | 
					
						
							|  |  |  |     int index = path.size() - 1; | 
					
						
							|  |  |  |     for (; index != -1; --index) { | 
					
						
							|  |  |  |         paramDecl = path.at(index)->asParameterDeclaration(); | 
					
						
							|  |  |  |         if (paramDecl) | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (index < 1) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ParameterDeclarationClauseAST *paramDeclClause = path.at(index-1)->asParameterDeclarationClause(); | 
					
						
							|  |  |  |     QTC_ASSERT(paramDeclClause && paramDeclClause->parameter_declaration_list, return); | 
					
						
							| 
									
										
										
										
											2012-10-10 23:27:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ParameterDeclarationListAST *paramListNode = paramDeclClause->parameter_declaration_list; | 
					
						
							|  |  |  |     ParameterDeclarationListAST *prevParamListNode = 0; | 
					
						
							|  |  |  |     while (paramListNode) { | 
					
						
							|  |  |  |         if (paramDecl == paramListNode->value) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         prevParamListNode = paramListNode; | 
					
						
							|  |  |  |         paramListNode = paramListNode->next; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-04 14:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     if (!paramListNode) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (prevParamListNode) | 
					
						
							|  |  |  |         result.append(CppQuickFixOperation::Ptr( | 
					
						
							|  |  |  |             new RearrangeParamDeclarationListOp(interface, paramListNode->value, | 
					
						
							|  |  |  |                 prevParamListNode->value, RearrangeParamDeclarationListOp::TargetPrevious))); | 
					
						
							|  |  |  |     if (paramListNode->next) | 
					
						
							|  |  |  |         result.append(CppQuickFixOperation::Ptr( | 
					
						
							|  |  |  |             new RearrangeParamDeclarationListOp(interface, paramListNode->value, | 
					
						
							|  |  |  |                 paramListNode->next->value, RearrangeParamDeclarationListOp::TargetNext))); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-12-21 18:24:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | class ReformatPointerDeclarationOp: public CppQuickFixOperation | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     ReformatPointerDeclarationOp(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                  const Utils::ChangeSet change) | 
					
						
							|  |  |  |         : CppQuickFixOperation(interface) | 
					
						
							|  |  |  |         , m_change(change) | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         QString description; | 
					
						
							|  |  |  |         if (m_change.operationList().size() == 1) { | 
					
						
							|  |  |  |             description = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                 "Reformat to \"%1\"").arg(m_change.operationList().first().text); | 
					
						
							|  |  |  |         } else { // > 1
 | 
					
						
							|  |  |  |             description = QApplication::translate("CppTools::QuickFix", | 
					
						
							|  |  |  |                 "Reformat Pointers or References"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         setDescription(description); | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     void perform() | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |         CppRefactoringChanges refactoring(snapshot()); | 
					
						
							|  |  |  |         CppRefactoringFilePtr currentFile = refactoring.file(fileName()); | 
					
						
							|  |  |  |         currentFile->setChangeSet(m_change); | 
					
						
							|  |  |  |         currentFile->apply(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     Utils::ChangeSet m_change; | 
					
						
							| 
									
										
										
										
											2013-01-17 17:23:43 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-12 11:57:01 +01:00
										 |  |  | /// Filter the results of ASTPath.
 | 
					
						
							|  |  |  | /// The resulting list contains the supported AST types only once.
 | 
					
						
							|  |  |  | /// For this, the results of ASTPath are iterated in reverse order.
 | 
					
						
							|  |  |  | class ReformatPointerDeclarationASTPathResultsFilter | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ReformatPointerDeclarationASTPathResultsFilter() | 
					
						
							|  |  |  |         : m_hasSimpleDeclaration(false) | 
					
						
							|  |  |  |         , m_hasFunctionDefinition(false) | 
					
						
							|  |  |  |         , m_hasParameterDeclaration(false) | 
					
						
							|  |  |  |         , m_hasIfStatement(false) | 
					
						
							|  |  |  |         , m_hasWhileStatement(false) | 
					
						
							|  |  |  |         , m_hasForStatement(false) | 
					
						
							|  |  |  |         , m_hasForeachStatement(false) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QList<AST*> filter(const QList<AST*> &astPathList) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QList<AST*> filtered; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int i = astPathList.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |             AST *ast = astPathList.at(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (! m_hasSimpleDeclaration && ast->asSimpleDeclaration()) { | 
					
						
							|  |  |  |                 m_hasSimpleDeclaration = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasFunctionDefinition && ast->asFunctionDefinition()) { | 
					
						
							|  |  |  |                 m_hasFunctionDefinition = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasParameterDeclaration && ast->asParameterDeclaration()) { | 
					
						
							|  |  |  |                 m_hasParameterDeclaration = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasIfStatement && ast->asIfStatement()) { | 
					
						
							|  |  |  |                 m_hasIfStatement = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasWhileStatement && ast->asWhileStatement()) { | 
					
						
							|  |  |  |                 m_hasWhileStatement = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasForStatement && ast->asForStatement()) { | 
					
						
							|  |  |  |                 m_hasForStatement = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } else if (! m_hasForeachStatement && ast->asForeachStatement()) { | 
					
						
							|  |  |  |                 m_hasForeachStatement = true; | 
					
						
							|  |  |  |                 filtered.append(ast); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return filtered; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     bool m_hasSimpleDeclaration; | 
					
						
							|  |  |  |     bool m_hasFunctionDefinition; | 
					
						
							|  |  |  |     bool m_hasParameterDeclaration; | 
					
						
							|  |  |  |     bool m_hasIfStatement; | 
					
						
							|  |  |  |     bool m_hasWhileStatement; | 
					
						
							|  |  |  |     bool m_hasForStatement; | 
					
						
							|  |  |  |     bool m_hasForeachStatement; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  | void ReformatPointerDeclaration::match(const CppQuickFixInterface &interface, | 
					
						
							|  |  |  |                                        QuickFixOperations &result) | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |     const QList<AST *> &path = interface->path(); | 
					
						
							|  |  |  |     CppRefactoringFilePtr file = interface->currentFile(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Overview overview = CppCodeStyleSettings::currentProjectCodeStyleOverview(); | 
					
						
							|  |  |  |     overview.showArgumentNames = true; | 
					
						
							|  |  |  |     overview.showReturnTypes = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QTextCursor cursor = file->cursor(); | 
					
						
							|  |  |  |     Utils::ChangeSet change; | 
					
						
							|  |  |  |     PointerDeclarationFormatter formatter(file, overview, | 
					
						
							|  |  |  |         PointerDeclarationFormatter::RespectCursor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cursor.hasSelection()) { | 
					
						
							|  |  |  |         // This will no work always as expected since this method is only called if
 | 
					
						
							|  |  |  |         // interface-path() is not empty. If the user selects the whole document via
 | 
					
						
							|  |  |  |         // ctrl-a and there is an empty line in the end, then the cursor is not on
 | 
					
						
							|  |  |  |         // any AST and therefore no quick fix will be triggered.
 | 
					
						
							|  |  |  |         change = formatter.format(file->cppDocument()->translationUnit()->ast()); | 
					
						
							|  |  |  |         if (! change.isEmpty()) { | 
					
						
							|  |  |  |             result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                 new ReformatPointerDeclarationOp(interface, change))); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2013-02-12 11:57:01 +01:00
										 |  |  |         const QList<AST *> suitableASTs | 
					
						
							|  |  |  |             = ReformatPointerDeclarationASTPathResultsFilter().filter(path); | 
					
						
							|  |  |  |         foreach (AST *ast, suitableASTs) { | 
					
						
							| 
									
										
										
										
											2013-02-07 12:16:41 +01:00
										 |  |  |             change = formatter.format(ast); | 
					
						
							|  |  |  |             if (! change.isEmpty()) { | 
					
						
							|  |  |  |                 result.append(QuickFixOperation::Ptr( | 
					
						
							|  |  |  |                     new ReformatPointerDeclarationOp(interface, change))); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-07-26 13:06:33 +02:00
										 |  |  | } |