All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.netbeans.modules.java.hints.UtilityClass Maven / Gradle / Ivy

There is a newer version: RELEASE240
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.netbeans.modules.java.hints;

import com.sun.source.tree.ClassTree;
import com.sun.source.tree.ExpressionTree;
import com.sun.source.tree.MethodTree;
import com.sun.source.tree.ModifiersTree;
import com.sun.source.tree.StatementTree;
import com.sun.source.tree.Tree;
import com.sun.source.tree.Tree.Kind;
import com.sun.source.tree.TypeParameterTree;
import com.sun.source.tree.VariableTree;
import com.sun.source.util.TreePath;
import java.util.Collections;
import javax.lang.model.element.Element;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.Name;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.ElementFilter;
import org.netbeans.api.java.source.CompilationInfo;
import org.netbeans.api.java.source.TreePathHandle;
import org.netbeans.api.java.source.TreeUtilities;
import org.netbeans.api.java.source.WorkingCopy;
import org.netbeans.spi.editor.hints.ErrorDescription;
import org.netbeans.spi.editor.hints.Severity;
import org.netbeans.spi.java.hints.ErrorDescriptionFactory;
import org.netbeans.spi.java.hints.Hint;
import org.netbeans.spi.java.hints.HintContext;
import org.netbeans.spi.java.hints.JavaFix;
import org.netbeans.spi.java.hints.TriggerTreeKind;
import org.openide.util.NbBundle;

/**
 *
 * @author Jaroslav tulach
 */
public class UtilityClass {
    
    @Hint(id="org.netbeans.modules.java.hints.UtilityClass_1", displayName="#MSG_UtilityClass", description="#HINT_UtilityClass", category="api", enabled=false, severity=Severity.VERIFIER, suppressWarnings="UtilityClassWithoutPrivateConstructor")
    @TriggerTreeKind(Kind.CLASS)
    public static ErrorDescription utilityClass(HintContext ctx) {
        CompilationInfo compilationInfo = ctx.getInfo();
        TreePath treePath = ctx.getPath();
        Element e = compilationInfo.getTrees().getElement(treePath);
        if (e == null) {
            return null;
        }
        
        if (!isUtilityClass(compilationInfo, e)) return null;
        
        for (ExecutableElement c : ElementFilter.constructorsIn(e.getEnclosedElements())) {
            if (!compilationInfo.getElementUtilities().isSynthetic(c)) {
                return null;
            }
        }

        return ErrorDescriptionFactory.forName(ctx,
                                               treePath,
                                               NbBundle.getMessage(UtilityClass.class, "MSG_UtilityClass"),
                                               new FixImpl(true,
                                                           TreePathHandle.create(e, compilationInfo)
                                               ).toEditorFix());
    }
    
    @Hint(id="org.netbeans.modules.java.hints.UtilityClass_2", displayName="#MSG_PublicConstructor", description="#HINT_PublicConstructor", category="api", enabled=false, severity=Severity.HINT, suppressWarnings="UtilityClassWithPublicConstructor")
    @TriggerTreeKind(Kind.METHOD)
    public static ErrorDescription constructor(HintContext ctx) {
        CompilationInfo compilationInfo = ctx.getInfo();
        TreePath treePath = ctx.getPath();
        Element e = compilationInfo.getTrees().getElement(treePath);
        if (e == null) {
            return null;
        }
        if (   e.getKind() != ElementKind.CONSTRUCTOR
            || compilationInfo.getElementUtilities().isSynthetic(e)
            || (!e.getModifiers().contains(Modifier.PROTECTED) && !e.getModifiers().contains(Modifier.PUBLIC))) {
            return null;
        }
        
        if (!isUtilityClass(compilationInfo, e.getEnclosingElement())) return null;
        
        return ErrorDescriptionFactory.forName(ctx,
                                               treePath,
                                               NbBundle.getMessage(UtilityClass.class, "MSG_PublicConstructor"),
                                               new FixImpl(false,
                                                           TreePathHandle.create(e, compilationInfo)
                                               ).toEditorFix());
    }
    
    private static boolean isMainMethod(ExecutableElement m) {
        return m.getModifiers().contains(Modifier.STATIC) &&
               m.getSimpleName().contentEquals("main") &&
               (m.getReturnType().getKind() == TypeKind.VOID) &&
               (m.getParameters().size() == 1) &&
               (m.getParameters().get(0).asType().toString().equals("java.lang.String[]"));
    }

    private static boolean isUtilityClass(CompilationInfo compilationInfo, Element clazz) {
        if (clazz.getKind() != ElementKind.CLASS) {
            return false;
        }
        
        TypeMirror supr = ((TypeElement) clazz).getSuperclass();
        if (supr == null) {
            return false;
        }
        Element superElem = compilationInfo.getTypes().asElement(supr);
        if (superElem instanceof TypeElement) {
            Name superName = compilationInfo.getElements().getBinaryName((TypeElement) superElem);
            if (superName != null && !superName.contentEquals("java.lang.Object")) {
                return false;
            }
        }

        int cnt = 0;
        for (Element m : clazz.getEnclosedElements()) {
            if (m.getKind() == ElementKind.METHOD && isMainMethod(((ExecutableElement) m))) return false;
            if (m.getKind() == ElementKind.METHOD || m.getKind() == ElementKind.FIELD) {
                if (!m.getModifiers().contains(Modifier.STATIC)) return false;
                cnt++;
            }
        }

        return cnt > 0;
    }
    
    private static final class FixImpl extends JavaFix {
        private boolean clazz;

        public FixImpl(boolean clazz, TreePathHandle handle) {
            super(handle);
            this.clazz = clazz;
        }

        public String getText() {
            return NbBundle.getMessage(UtilityClass.class, clazz ? "MSG_PrivateConstructor" : "MSG_MakePrivate"); // NOI18N
        }

        @Override public String toString() {
            return "FixUtilityClass"; // NOI18N
        }

        @Override
        protected void performRewrite(TransformationContext ctx) {
            WorkingCopy wc = ctx.getWorkingCopy();
            TreePath tp = ctx.getPath();
            Element e = wc.getTrees().getElement(tp);
            if (e == null) {
                return;
            }
            Tree outer = wc.getTrees().getTree(e);
            if (clazz) {
                if (outer == null || !TreeUtilities.CLASS_TREE_KINDS.contains(outer.getKind())) {
                    return;
                }
                ClassTree cls = (ClassTree)outer;

                ModifiersTree modifiers = wc.getTreeMaker().Modifiers(Collections.singleton(Modifier.PRIVATE));
                MethodTree m = wc.getTreeMaker().Constructor(
                    modifiers,
                    Collections.emptyList(),
                    Collections.emptyList(),
                    Collections.emptyList(),
                    wc.getTreeMaker().Block(Collections.emptyList(), false)
                );
                wc.rewrite(cls, wc.getTreeMaker().addClassMember(cls, m));
            } else {
                if (outer == null || outer.getKind() != Kind.METHOD) {
                    return;
                }
                MethodTree met = (MethodTree)outer;

                ModifiersTree modifiers = wc.getTreeMaker().Modifiers(Collections.singleton(Modifier.PRIVATE), met.getModifiers().getAnnotations());
                wc.rewrite(met.getModifiers(), modifiers);
            }
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy