/******************************************************************************* * Copyright (c) 2008, 2016 Institute for Software, HSR Hochschule fuer Technik * Rapperswil, University of applied sciences and others * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Institute for Software - initial API and implementation * Sergey Prigogin (Google) *******************************************************************************/ package org.eclipse.cdt.internal.ui.refactoring.utils; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.Path; import org.eclipse.cdt.core.dom.ast.IASTDeclSpecifier; import org.eclipse.cdt.core.dom.ast.IASTDeclaration; import org.eclipse.cdt.core.dom.ast.IASTDeclarator; import org.eclipse.cdt.core.dom.ast.IASTFunctionDeclarator; import org.eclipse.cdt.core.dom.ast.IASTFunctionDefinition; import org.eclipse.cdt.core.dom.ast.IASTName; import org.eclipse.cdt.core.dom.ast.IASTNode; import org.eclipse.cdt.core.dom.ast.IASTSimpleDeclaration; import org.eclipse.cdt.core.dom.ast.IASTTranslationUnit; import org.eclipse.cdt.core.dom.ast.IBinding; import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTCompositeTypeSpecifier; import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTDeclSpecifier; import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTFunctionDeclarator; import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTQualifiedName; import org.eclipse.cdt.core.dom.ast.cpp.ICPPASTTemplateDeclaration; import org.eclipse.cdt.core.dom.ast.cpp.ICPPMethod; import org.eclipse.cdt.internal.core.dom.parser.ASTQueries; import org.eclipse.cdt.internal.core.dom.parser.cpp.CPPASTNamespaceDefinition; import org.eclipse.cdt.internal.core.dom.parser.cpp.CPPASTTranslationUnit; import org.eclipse.cdt.internal.core.dom.parser.cpp.semantics.CPPVisitor; import org.eclipse.cdt.internal.ui.refactoring.CRefactoringContext; import org.eclipse.cdt.internal.ui.refactoring.MethodContext; /** * General class for common Node operations. * * @author Lukas Felber & Guido Zgraggen */ public class NodeHelper { public static IASTDeclaration[] getDeclarations(IASTNode parent) { if (parent instanceof ICPPASTCompositeTypeSpecifier) { return ((ICPPASTCompositeTypeSpecifier) parent).getMembers(); } else if (parent instanceof CPPASTTranslationUnit) { return ((CPPASTTranslationUnit) parent).getDeclarations(); } else if (parent instanceof CPPASTNamespaceDefinition) { return ((CPPASTNamespaceDefinition) parent).getDeclarations(); } return IASTDeclaration.EMPTY_DECLARATION_ARRAY; } public static IASTNode findFollowingNode(IASTNode currentNode) { if (currentNode == null || currentNode.getParent() == null) { return null; } boolean match = false; for (IASTNode actNode : getDeclarations(currentNode.getParent())) { if (match) { return actNode; } if (actNode.equals(currentNode)) { match = true; } } return null; } public static IASTNode findTopLevelParent(IASTNode currentNode) { while (currentNode != null && currentNode.getParent() != null && currentNode.getParent().getParent() != null) { return findTopLevelParent(currentNode.getParent()); } return currentNode; } public static boolean isSameNode(IASTNode node1, IASTNode node2) { if (node1 == null || node2 == null) { return false; } return node1.getNodeLocations()[0].getNodeOffset() == node2.getNodeLocations()[0].getNodeOffset() && node1.getNodeLocations()[0].getNodeLength() == node2.getNodeLocations()[0].getNodeLength() && new Path(node1.getFileLocation().getFileName()).equals(new Path(node2.getFileLocation().getFileName())); } public static MethodContext findMethodContext(IASTNode node, CRefactoringContext refactoringContext, IProgressMonitor pm) throws CoreException { IASTTranslationUnit translationUnit = node.getTranslationUnit(); boolean found = false; MethodContext context = new MethodContext(); IASTName name = null; while (node != null && !found) { node = node.getParent(); if (node instanceof IASTFunctionDeclarator) { name = ((IASTFunctionDeclarator) node).getName(); found = true; context.setType(MethodContext.ContextType.FUNCTION); } else if (node instanceof IASTFunctionDefinition) { name = CPPVisitor.findInnermostDeclarator(((IASTFunctionDefinition) node).getDeclarator()).getName(); found = true; context.setType(MethodContext.ContextType.FUNCTION); } } if (name != null) { getMethodContexWithIndex(refactoringContext, translationUnit, name, context, pm); } return context; } private static void getMethodContexWithIndex(CRefactoringContext refactoringContext, IASTTranslationUnit ast, IASTName name, MethodContext context, IProgressMonitor pm) throws CoreException { if (name instanceof ICPPASTQualifiedName) { ICPPASTQualifiedName qname = (ICPPASTQualifiedName) name; context.setMethodQName(qname); } IBinding binding = name.resolveBinding(); if (binding instanceof ICPPMethod) { context.setType(MethodContext.ContextType.METHOD); IASTName declName = DefinitionFinder.getMemberDeclaration(name, refactoringContext, pm); context.setMethodDeclarationName(declName); } } public static boolean isMethodDeclaration(IASTSimpleDeclaration simpleDeclaration) { if (simpleDeclaration == null) { return false; } final IASTDeclSpecifier declSpecifier = simpleDeclaration.getDeclSpecifier(); final IASTDeclarator[] declarators = simpleDeclaration.getDeclarators(); if ((declSpecifier instanceof ICPPASTDeclSpecifier) && ((ICPPASTDeclSpecifier) declSpecifier).isFriend()) { return false; } return declarators.length == 1 && declarators[0] instanceof ICPPASTFunctionDeclarator; } public static boolean isContainedInTemplateDeclaration(IASTNode node) { return ASTQueries.findAncestorWithType(node, ICPPASTTemplateDeclaration.class) != null; } }