blob: bfe1fc80b9c75b60749abd790292f8e42eb79308 [file] [log] [blame]
/*
* 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.j2ee.jpa.refactoring;
import com.sun.source.tree.CompilationUnitTree;
import com.sun.source.util.TreePath;
import java.io.File;
import java.io.IOException;
import java.net.URISyntaxException;
import java.net.URL;
import java.net.URLDecoder;
import javax.lang.model.element.Element;
import javax.lang.model.element.TypeElement;
import org.netbeans.api.java.classpath.ClassPath;
import org.netbeans.api.java.project.JavaProjectConstants;
import org.netbeans.api.java.source.CancellableTask;
import org.netbeans.api.java.source.ClasspathInfo;
import org.netbeans.api.java.source.CompilationController;
import org.netbeans.api.java.source.CompilationInfo;
import org.netbeans.api.java.source.JavaSource;
import org.netbeans.api.java.source.TreePathHandle;
import org.netbeans.api.project.FileOwnerQuery;
import org.netbeans.api.project.Project;
import org.netbeans.api.project.ProjectUtils;
import org.netbeans.api.project.SourceGroup;
import org.netbeans.api.project.ui.OpenProjects;
import org.netbeans.modules.refactoring.api.AbstractRefactoring;
import org.netbeans.modules.refactoring.api.Problem;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileUtil;
import org.openide.util.Exceptions;
import org.openide.util.Parameters;
import org.openide.util.Utilities;
/**
* Utility methods needed in JPA refactoring, split from
* <code>org.netbeans.modules.j2ee.refactoring.Utility</code>.
*
* @author Erno Mononen
*/
public abstract class RefactoringUtil {
private static final String JAVA_MIME_TYPE = "text/x-java"; //NO18N
/**
* Constructs a new fully qualified name for the given <code>newName</code>.
*
* @param originalFullyQualifiedName the old fully qualified name of the class.
* @param newName the new unqualified name of the class.
*
* @return the new fully qualified name of the class.
*/
public static String renameClass(String originalFullyQualifiedName, String newName){
Parameters.notEmpty("originalFullyQualifiedName", originalFullyQualifiedName); //NO18N
Parameters.notEmpty("newName", newName); //NO18N
int lastDot = originalFullyQualifiedName.lastIndexOf('.');
return (lastDot <= 0) ? newName : originalFullyQualifiedName.substring(0, lastDot + 1) + newName;
}
/**
* Sets the given <code>toAdd</code> as the following problem for
* the given <code>existing</code> problem.
*
* @param toAdd the problem to add, may be null.
* @param existing the problem whose following problem should be set, may be null.
*
* @return the existing problem with its following problem
* set to the given problem or null if both of the params
* were null.
*
*/
public static Problem addToEnd(Problem toAdd, Problem existing){
if (existing == null){
return toAdd;
}
if (toAdd == null){
return existing;
}
Problem tail = existing;
while(tail.getNext() != null){
tail = tail.getNext();
}
tail.setNext(toAdd);
return tail;
}
// copied from o.n.m.java.refactoring.RetoucheUtils
public static boolean isOnSourceClasspath(FileObject fo) {
Project p = FileOwnerQuery.getOwner(fo);
if (p==null) {
return false;
}
if (OpenProjects.getDefault().isProjectOpen(p)) {
SourceGroup[] gr = ProjectUtils.getSources(p).getSourceGroups(JavaProjectConstants.SOURCES_TYPE_JAVA);
for (int j = 0; j < gr.length; j++) {
if (fo==gr[j].getRootFolder()) {
return true;
}
if (FileUtil.isParentOf(gr[j].getRootFolder(), fo)) {
return true;
}
}
return false;
}
return false;
}
// copied from o.n.m.java.refactoring.RetoucheUtils
public static boolean isJavaFile(FileObject f) {
return JAVA_MIME_TYPE.equals(f.getMIMEType()); //NOI18N
}
/**
* Gets the name of the property associated with the given accessor.
*
* @param accessor the name of the accessor method of the property. Must follow the JavaBeans
* naming conventions, i.e. start with 'get/set/is' followed by an uppercase letter,
* otherwise it is assumed that the name of the property directly matches with
* the getter. Must not be null or empty.
*
* @return the property name resolved from the given <code>getter</code>, i.e.
* if the given arg was <code>getProperty</code>, this method will return
* <code>property</code>.
*/
public static String getPropertyName(String accessor){
Parameters.notEmpty("accessor", accessor); //NO18N
int prefixLength = getPrefixLength(accessor);
String withoutPrefix = accessor.substring(prefixLength);
char firstChar = withoutPrefix.charAt(0);
if (!Character.isUpperCase(firstChar)){
return accessor;
}
return Character.toLowerCase(firstChar) + withoutPrefix.substring(1);
}
private static int getPrefixLength(String accessor){
String[] accessorPrefixes = new String[]{"get", "set", "is"}; //NO18N
for (String prefix : accessorPrefixes){
if (accessor.startsWith(prefix)){
return prefix.length();
}
}
return 0;
}
/**
* Resolves the TreePathHandle for the given refactoring.
* @param refactoring the refactoring.
* @return the TreePathHandle or null if no handle could be resolved.
*/
public static TreePathHandle resolveTreePathHandle(AbstractRefactoring refactoring) throws IOException {
Parameters.notNull("refactoring", refactoring); //NO18N
TreePathHandle tph = refactoring.getRefactoringSource().lookup(TreePathHandle.class);
if (tph != null) {
return tph;
}
final TreePathHandle[] result = new TreePathHandle[1];
JavaSource source = JavaSource.forFileObject(refactoring.getRefactoringSource().lookup(FileObject.class));
source.runUserActionTask(new CancellableTask<CompilationController>() {
@Override
public void cancel() {
}
@Override
public void run(CompilationController co) throws Exception {
co.toPhase(JavaSource.Phase.RESOLVED);
CompilationUnitTree cut = co.getCompilationUnit();
result[0] = TreePathHandle.create(TreePath.getPath(cut, cut.getTypeDecls().get(0)), co);
}
}, true);
return result[0];
}
/**
* Gets a TreePathHandle for the specified property.
*
* @param fieldName the name of the field.
* @param className the FQN of the class.
* @param file the file object representing the class.
*/
public static TreePathHandle getTreePathHandle(final String fieldName,
final String className, FileObject file) throws IOException{
final TreePathHandle[] result = new TreePathHandle[1];
JavaSource source = JavaSource.forFileObject(file);
source.runUserActionTask(new CancellableTask<CompilationController>() {
@Override
public void cancel() {
}
@Override
public void run(CompilationController info) throws Exception {
info.toPhase(JavaSource.Phase.RESOLVED);
TypeElement te = info.getElements().getTypeElement(className);
for (Element enclosed : te.getEnclosedElements()){
if (enclosed.getSimpleName().contentEquals(fieldName)){
TreePath propertyPath = info.getTrees().getPath(enclosed);
result[0] = TreePathHandle.create(propertyPath, info);
}
}
}
}, true);
return result[0];
}
// copied from o.n.m.java.refactoring.RetoucheUtils
// XXX use j2ee/core/utilities
public static String getPackageName(URL url) {
File f = null;
try {
f = FileUtil.normalizeFile(Utilities.toFile(url.toURI()));
} catch (URISyntaxException uRISyntaxException) {
throw new IllegalArgumentException("Cannot create package name for url " + url);
}
String suffix = "";
do {
FileObject fo = FileUtil.toFileObject(f);
if (fo != null) {
if ("".equals(suffix)) {
return getPackageName(fo);
}
String prefix = getPackageName(fo);
return prefix + ("".equals(prefix)?"":".") + suffix;
}
if (!"".equals(suffix)) {
suffix = "." + suffix;
}
suffix = URLDecoder.decode(f.getPath().substring(f.getPath().lastIndexOf(File.separatorChar)+1)) + suffix;
f = f.getParentFile();
} while (f!=null);
throw new IllegalArgumentException("Cannot create package name for url " + url);
}
// copied from o.n.m.java.refactoring.RetoucheUtils
// XXX use j2ee/core/utilities
private static String getPackageName(FileObject folder) {
assert folder.isFolder() : "argument must be folder";
return ClassPath.getClassPath(
folder, ClassPath.SOURCE)
.getResourceName(folder, '.', false);
}
}