/******************************************************************************* * Copyright (c) 2000, 2011 IBM Corporation 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: * IBM Corporation - initial API and implementation * Mateusz Wenus <mateusz.wenus@gmail.com> - [override method] generate in declaration order [code generation] - https://bugs.eclipse * .org/bugs/show_bug.cgi?id=140971 *******************************************************************************/ package org.eclipse.che.ide.ext.java.jdt.internal.corext.fix; import org.eclipse.che.ide.ext.java.jdt.JavaPreferencesSettings; import org.eclipse.che.ide.ext.java.jdt.core.dom.ASTNode; import org.eclipse.che.ide.ext.java.jdt.core.dom.AbstractTypeDeclaration; import org.eclipse.che.ide.ext.java.jdt.core.dom.AnonymousClassDeclaration; import org.eclipse.che.ide.ext.java.jdt.core.dom.CompilationUnit; import org.eclipse.che.ide.ext.java.jdt.core.dom.EnumConstantDeclaration; import org.eclipse.che.ide.ext.java.jdt.core.dom.EnumDeclaration; import org.eclipse.che.ide.ext.java.jdt.core.dom.IMethodBinding; import org.eclipse.che.ide.ext.java.jdt.core.dom.ITypeBinding; import org.eclipse.che.ide.ext.java.jdt.core.dom.IVariableBinding; import org.eclipse.che.ide.ext.java.jdt.core.dom.MethodDeclaration; import org.eclipse.che.ide.ext.java.jdt.core.dom.rewrite.ASTRewrite; import org.eclipse.che.ide.ext.java.jdt.core.dom.rewrite.ImportRewrite; import org.eclipse.che.ide.ext.java.jdt.core.dom.rewrite.ImportRewrite.ImportRewriteContext; import org.eclipse.che.ide.ext.java.jdt.core.dom.rewrite.ListRewrite; import org.eclipse.che.ide.ext.java.jdt.internal.corext.codemanipulation.CodeGenerationSettings; import org.eclipse.che.ide.ext.java.jdt.internal.corext.codemanipulation.ContextSensitiveImportRewriteContext; import org.eclipse.che.ide.ext.java.jdt.internal.corext.codemanipulation.StubUtility2; import org.eclipse.che.ide.ext.java.jdt.internal.corext.fix.CompilationUnitRewriteOperationsFix.CompilationUnitRewriteOperation; import org.eclipse.che.ide.ext.java.jdt.internal.corext.refactoring.structure.CompilationUnitRewrite; import org.eclipse.che.ide.ext.java.jdt.internal.corext.util.MethodsSourcePositionComparator; import org.eclipse.che.ide.ext.java.jdt.internal.text.correction.CorrectionMessages; import org.eclipse.che.ide.runtime.Assert; import org.eclipse.che.ide.runtime.CoreException; import java.util.Arrays; public class AddUnimplementedMethodsOperation extends CompilationUnitRewriteOperation { private ASTNode fTypeNode; /** * Create a {@link AddUnimplementedMethodsOperation} * * @param typeNode * must be one of the following types: * <ul><li>AnonymousClassDeclaration</li> * <li>AbstractTypeDeclaration</li> * <li>EnumConstantDeclaration</li></ul> */ public AddUnimplementedMethodsOperation(ASTNode typeNode) { fTypeNode = typeNode; } /** {@inheritDoc} */ @Override public void rewriteAST(CompilationUnitRewrite cuRewrite) throws CoreException { IMethodBinding[] unimplementedMethods = getUnimplementedMethods(fTypeNode); if (unimplementedMethods.length == 0) return; ImportRewriteContext context = new ContextSensitiveImportRewriteContext((CompilationUnit)fTypeNode.getRoot(), fTypeNode.getStartPosition(), cuRewrite.getImportRewrite()); ASTRewrite rewrite = cuRewrite.getASTRewrite(); // ICompilationUnit unit= cuRewrite.getCu(); CodeGenerationSettings settings = JavaPreferencesSettings.getCodeGenerationSettings(); ListRewrite listRewrite; if (fTypeNode instanceof AnonymousClassDeclaration) { AnonymousClassDeclaration decl = (AnonymousClassDeclaration)fTypeNode; listRewrite = rewrite.getListRewrite(decl, AnonymousClassDeclaration.BODY_DECLARATIONS_PROPERTY); settings.createComments = false; } else if (fTypeNode instanceof AbstractTypeDeclaration) { AbstractTypeDeclaration decl = (AbstractTypeDeclaration)fTypeNode; listRewrite = rewrite.getListRewrite(decl, decl.getBodyDeclarationsProperty()); } else if (fTypeNode instanceof EnumConstantDeclaration) { EnumConstantDeclaration enumConstantDeclaration = (EnumConstantDeclaration)fTypeNode; AnonymousClassDeclaration anonymousClassDeclaration = enumConstantDeclaration.getAnonymousClassDeclaration(); if (anonymousClassDeclaration == null) { anonymousClassDeclaration = rewrite.getAST().newAnonymousClassDeclaration(); rewrite.set( enumConstantDeclaration, EnumConstantDeclaration.ANONYMOUS_CLASS_DECLARATION_PROPERTY, anonymousClassDeclaration, createTextEditGroup( CorrectionMessages.INSTANCE.AddUnimplementedMethodsOperation_AddMissingMethod_group(), cuRewrite)); } listRewrite = rewrite.getListRewrite(anonymousClassDeclaration, AnonymousClassDeclaration.BODY_DECLARATIONS_PROPERTY); settings.createComments = false; } else { Assert.isTrue(false, "Unknown type node"); //$NON-NLS-1$ return; } ImportRewrite imports = cuRewrite.getImportRewrite(); for (int i = 0; i < unimplementedMethods.length; i++) { IMethodBinding curr = unimplementedMethods[i]; MethodDeclaration newMethodDecl = StubUtility2.createImplementationStub(rewrite, imports, context, curr, curr.getDeclaringClass().getName(), settings, false); listRewrite.insertLast( newMethodDecl, createTextEditGroup(CorrectionMessages.INSTANCE.AddUnimplementedMethodsOperation_AddMissingMethod_group(), cuRewrite)); } } /** {@inheritDoc} */ @Override public String getAdditionalInfo() { if (fTypeNode instanceof EnumDeclaration) return CorrectionMessages.INSTANCE.UnimplementedMethodsCorrectionProposal_enum_info(); IMethodBinding[] methodsToOverride = getMethodsToImplement(); StringBuffer buf = new StringBuffer(); buf.append("<b>"); //$NON-NLS-1$ if (methodsToOverride.length == 1) { buf.append(CorrectionMessages.INSTANCE.UnimplementedMethodsCorrectionProposal_info_singular()); } else { buf.append(CorrectionMessages.INSTANCE.UnimplementedMethodsCorrectionProposal_info_plural(String .valueOf( methodsToOverride .length))); } buf.append("</b><ul>"); //$NON-NLS-1$ for (int i = 0; i < methodsToOverride.length; i++) { buf.append("<li>"); //$NON-NLS-1$ //TODO // buf.append(BindingLabelProvider.getBindingLabel(methodsToOverride[i], JavaElementLabels.ALL_FULLY_QUALIFIED)); buf.append(methodsToOverride[0].getName()); buf.append("</li>"); //$NON-NLS-1$ } buf.append("</ul>"); //$NON-NLS-1$ return buf.toString(); } public IMethodBinding[] getMethodsToImplement() { return getUnimplementedMethods(fTypeNode); } private IMethodBinding[] getUnimplementedMethods(ASTNode typeNode) { ITypeBinding binding = null; boolean implementAbstractsOfInput = false; if (typeNode instanceof AnonymousClassDeclaration) { AnonymousClassDeclaration decl = (AnonymousClassDeclaration)typeNode; binding = decl.resolveBinding(); } else if (typeNode instanceof AbstractTypeDeclaration) { AbstractTypeDeclaration decl = (AbstractTypeDeclaration)typeNode; binding = decl.resolveBinding(); } else if (typeNode instanceof EnumConstantDeclaration) { EnumConstantDeclaration enumConstantDeclaration = (EnumConstantDeclaration)typeNode; if (enumConstantDeclaration.getAnonymousClassDeclaration() != null) { binding = enumConstantDeclaration.getAnonymousClassDeclaration().resolveBinding(); } else { IVariableBinding varBinding = enumConstantDeclaration.resolveVariable(); if (varBinding != null) { binding = varBinding.getDeclaringClass(); implementAbstractsOfInput = true; } } } if (binding == null) return new IMethodBinding[0]; IMethodBinding[] unimplementedMethods = StubUtility2.getUnimplementedMethods(binding, implementAbstractsOfInput); Arrays.sort(unimplementedMethods, new MethodsSourcePositionComparator(binding)); return unimplementedMethods; } }