| /******************************************************************************* |
| * Copyright (c) 2007, 2010 Oracle. 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: |
| * Oracle - initial API and implementation |
| ******************************************************************************/ |
| package org.eclipse.jpt.core.internal.resource.java.source; |
| |
| import java.util.Iterator; |
| import java.util.List; |
| import org.eclipse.core.resources.IFile; |
| import org.eclipse.jdt.core.ICompilationUnit; |
| import org.eclipse.jdt.core.JavaModelException; |
| import org.eclipse.jdt.core.dom.ASTNode; |
| import org.eclipse.jdt.core.dom.AbstractTypeDeclaration; |
| import org.eclipse.jdt.core.dom.CompilationUnit; |
| import org.eclipse.jdt.core.dom.TypeDeclaration; |
| import org.eclipse.jpt.core.JpaAnnotationProvider; |
| import org.eclipse.jpt.core.JpaResourceModelListener; |
| import org.eclipse.jpt.core.JpaResourceType; |
| import org.eclipse.jpt.core.JptCorePlugin; |
| import org.eclipse.jpt.core.internal.utility.jdt.ASTTools; |
| import org.eclipse.jpt.core.resource.java.JavaResourceCompilationUnit; |
| import org.eclipse.jpt.core.resource.java.JavaResourcePersistentType; |
| import org.eclipse.jpt.core.utility.TextRange; |
| import org.eclipse.jpt.core.utility.jdt.AnnotationEditFormatter; |
| import org.eclipse.jpt.utility.CommandExecutor; |
| import org.eclipse.jpt.utility.internal.ListenerList; |
| import org.eclipse.jpt.utility.internal.iterators.EmptyIterator; |
| |
| /** |
| * Java compilation unit (source file) |
| */ |
| public final class SourceCompilationUnit |
| extends SourceNode |
| implements JavaResourceCompilationUnit |
| { |
| /** JDT compilation unit */ |
| private final ICompilationUnit compilationUnit; |
| |
| /** pluggable annotation provider */ |
| private final JpaAnnotationProvider annotationProvider; |
| |
| /** improved annotation formatting */ |
| private final AnnotationEditFormatter annotationEditFormatter; |
| |
| /** pluggable executor that allows the document to be modified on another thread */ |
| private final CommandExecutor modifySharedDocumentCommandExecutor; |
| |
| /** listeners notified whenever the resource model changes */ |
| private final ListenerList<JpaResourceModelListener> resourceModelListenerList; |
| |
| /** |
| * The primary type of the AST compilation unit. We are not going to handle |
| * multiple types defined in a single compilation unit. Entities must have |
| * a public/protected no-arg constructor, and there is no way to access |
| * the constructor in a package class (which is what all top-level, |
| * non-primary classes must be). |
| */ |
| private JavaResourcePersistentType persistentType; |
| |
| |
| // ********** construction ********** |
| |
| public SourceCompilationUnit( |
| ICompilationUnit compilationUnit, |
| JpaAnnotationProvider annotationProvider, |
| AnnotationEditFormatter annotationEditFormatter, |
| CommandExecutor modifySharedDocumentCommandExecutor) { |
| super(null); // the JPA compilation unit is the root of its sub-tree |
| this.compilationUnit = compilationUnit; |
| this.annotationProvider = annotationProvider; |
| this.annotationEditFormatter = annotationEditFormatter; |
| this.modifySharedDocumentCommandExecutor = modifySharedDocumentCommandExecutor; |
| this.resourceModelListenerList = new ListenerList<JpaResourceModelListener>(JpaResourceModelListener.class); |
| this.persistentType = this.buildPersistentType(); |
| } |
| |
| public void initialize(CompilationUnit astRoot) { |
| // never called? |
| } |
| |
| private JavaResourcePersistentType buildPersistentType() { |
| this.openCompilationUnit(); |
| CompilationUnit astRoot = this.buildASTRoot(); |
| this.closeCompilationUnit(); |
| return this.buildPersistentType(astRoot); |
| } |
| |
| private void openCompilationUnit() { |
| try { |
| this.compilationUnit.open(null); |
| } catch (JavaModelException ex) { |
| // do nothing - we just won't have a primary type in this case |
| } |
| } |
| |
| private void closeCompilationUnit() { |
| try { |
| this.compilationUnit.close(); |
| } catch (JavaModelException ex) { |
| // hmmm |
| } |
| } |
| |
| |
| // ********** AbstractJavaResourceNode overrides ********** |
| |
| @Override |
| protected boolean requiresParent() { |
| return false; |
| } |
| |
| @Override |
| public JavaResourceCompilationUnit getRoot() { |
| return this; |
| } |
| |
| @Override |
| public IFile getFile() { |
| return (IFile) this.compilationUnit.getResource(); |
| } |
| |
| @Override |
| public JpaAnnotationProvider getAnnotationProvider() { |
| return this.annotationProvider; |
| } |
| |
| |
| // ********** JavaResourceNode implementation ********** |
| |
| public void synchronizeWith(CompilationUnit astRoot) { |
| this.syncPersistentType(astRoot); |
| } |
| |
| public TextRange getTextRange(CompilationUnit astRoot) { |
| return null; |
| } |
| |
| |
| // ********** JavaResourceNode.Root implementation ********** |
| |
| /** |
| * NB: return *all* the persistent types since we build them all |
| */ |
| public Iterator<JavaResourcePersistentType> persistentTypes() { |
| return (this.persistentType == null) ? |
| EmptyIterator.<JavaResourcePersistentType>instance() : |
| this.persistentType.allTypes(); |
| } |
| |
| public void resourceModelChanged() { |
| for (JpaResourceModelListener listener : this.resourceModelListenerList.getListeners()) { |
| listener.resourceModelChanged(this); |
| } |
| } |
| |
| |
| // ********** JavaResourceCompilationUnit implementation ********** |
| |
| public ICompilationUnit getCompilationUnit() { |
| return this.compilationUnit; |
| } |
| |
| public void resolveTypes() { |
| if (this.persistentType != null) { |
| this.persistentType.resolveTypes(this.buildASTRoot()); |
| } |
| } |
| |
| public CommandExecutor getModifySharedDocumentCommandExecutor() { |
| return this.modifySharedDocumentCommandExecutor; |
| } |
| |
| public AnnotationEditFormatter getAnnotationEditFormatter() { |
| return this.annotationEditFormatter; |
| } |
| |
| public CompilationUnit buildASTRoot() { |
| return ASTTools.buildASTRoot(this.compilationUnit); |
| } |
| |
| |
| // ********** persistent type ********** |
| |
| private JavaResourcePersistentType buildPersistentType(CompilationUnit astRoot) { |
| TypeDeclaration td = this.getPrimaryTypeDeclaration(astRoot); |
| return (td == null) ? null : this.buildPersistentType(astRoot, td); |
| } |
| |
| |
| private void syncPersistentType(CompilationUnit astRoot) { |
| TypeDeclaration td = this.getPrimaryTypeDeclaration(astRoot); |
| if (td == null) { |
| this.syncPersistentType_(null); |
| } else { |
| if (this.persistentType == null) { |
| this.syncPersistentType_(this.buildPersistentType(astRoot, td)); |
| } else { |
| this.persistentType.synchronizeWith(astRoot); |
| } |
| } |
| } |
| |
| private void syncPersistentType_(JavaResourcePersistentType astPersistentType) { |
| JavaResourcePersistentType old = this.persistentType; |
| this.persistentType = astPersistentType; |
| this.firePropertyChanged(PERSISTENT_TYPES_COLLECTION, old, astPersistentType); |
| } |
| |
| |
| // ********** JpaResourceModel implementation ********** |
| |
| public JpaResourceType getResourceType() { |
| return JptCorePlugin.JAVA_SOURCE_RESOURCE_TYPE; |
| } |
| |
| public void addResourceModelListener(JpaResourceModelListener listener) { |
| this.resourceModelListenerList.add(listener); |
| } |
| |
| public void removeResourceModelListener(JpaResourceModelListener listener) { |
| this.resourceModelListenerList.remove(listener); |
| } |
| |
| |
| // ********** Java changes ********** |
| |
| public void synchronizeWithJavaSource() { |
| this.synchronizeWith(this.buildASTRoot()); |
| } |
| |
| |
| // ********** internal ********** |
| |
| private JavaResourcePersistentType buildPersistentType(CompilationUnit astRoot, TypeDeclaration typeDeclaration) { |
| return SourcePersistentType.newInstance(this, typeDeclaration, astRoot); |
| } |
| |
| /** |
| * i.e. the type with the same name as the compilation unit; |
| * return the first class or interface (ignore annotations and enums) with |
| * the same name as the compilation unit (file); |
| * NB: this type could be in error if there is an annotation or enum |
| * with the same name preceding it in the compilation unit |
| * |
| * Return null if the parser did not resolve the type declaration's binding. |
| * This can occur if the project JRE is removed (bug 225332). |
| */ |
| private TypeDeclaration getPrimaryTypeDeclaration(CompilationUnit astRoot) { |
| String primaryTypeName = this.getPrimaryTypeName(); |
| for (AbstractTypeDeclaration atd : this.types(astRoot)) { |
| if (this.nodeIsPrimaryTypeDeclaration(atd, primaryTypeName)) { |
| return (atd.resolveBinding() == null) ? null : (TypeDeclaration) atd; |
| } |
| } |
| return null; |
| } |
| |
| private boolean nodeIsPrimaryTypeDeclaration(AbstractTypeDeclaration atd, String primaryTypeName) { |
| return (atd.getNodeType() == ASTNode.TYPE_DECLARATION) && |
| atd.getName().getFullyQualifiedName().equals(primaryTypeName); |
| } |
| |
| // minimize scope of suppressed warnings |
| @SuppressWarnings("unchecked") |
| private List<AbstractTypeDeclaration> types(CompilationUnit astRoot) { |
| return astRoot.types(); |
| } |
| |
| /** |
| * i.e. the name of the compilation unit |
| */ |
| private String getPrimaryTypeName() { |
| return this.removeJavaExtension(this.compilationUnit.getElementName()); |
| } |
| |
| private String removeJavaExtension(String fileName) { |
| int index = fileName.lastIndexOf(".java"); //$NON-NLS-1$ |
| return (index == -1) ? fileName : fileName.substring(0, index); |
| } |
| |
| @Override |
| public void toString(StringBuilder sb) { |
| sb.append(this.getPrimaryTypeName()); |
| } |
| |
| } |