/******************************************************************************* * Copyright (c) 2000, 2008 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 *******************************************************************************/ package org.eclipse.jdt.internal.core; import java.util.HashMap; import org.eclipse.core.resources.IResource; import org.eclipse.core.runtime.IPath; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.jdt.core.IAnnotation; import org.eclipse.jdt.core.IBuffer; import org.eclipse.jdt.core.ICompilationUnit; import org.eclipse.jdt.core.IJavaElement; import org.eclipse.jdt.core.IMember; import org.eclipse.jdt.core.IOpenable; import org.eclipse.jdt.core.IParent; import org.eclipse.jdt.core.ISourceManipulation; import org.eclipse.jdt.core.ISourceRange; import org.eclipse.jdt.core.ISourceReference; import org.eclipse.jdt.core.JavaModelException; import org.eclipse.jdt.core.WorkingCopyOwner; import org.eclipse.jdt.core.dom.ASTNode; import org.eclipse.jdt.core.dom.CompilationUnit; import org.eclipse.jdt.internal.core.util.DOMFinder; import org.eclipse.jdt.internal.core.util.MementoTokenizer; import org.eclipse.jdt.internal.core.util.Messages; /** * Abstract class for Java elements which implement ISourceReference. */ public abstract class SourceRefElement extends JavaElement implements ISourceReference { /* * A count to uniquely identify this element in the case * that a duplicate named element exists. For example, if * there are two fields in a compilation unit with the * same name, the occurrence count is used to distinguish * them. The occurrence count starts at 1 (thus the first * occurrence is occurrence 1, not occurrence 0). */ public int occurrenceCount= 1; protected SourceRefElement(JavaElement parent) { super(parent); } /** * This element is being closed. Do any necessary cleanup. */ protected void closing(Object info) throws JavaModelException { // Do any necessary cleanup } /** * Returns a new element info for this element. */ protected Object createElementInfo() { return null; // not used for source ref elements } /** * @see ISourceManipulation */ public void copy(IJavaElement container, IJavaElement sibling, String rename, boolean force, IProgressMonitor monitor) throws JavaModelException { if (container == null) { throw new IllegalArgumentException(Messages.operation_nullContainer); } IJavaElement[] elements= new IJavaElement[] { this }; IJavaElement[] containers= new IJavaElement[] { container }; IJavaElement[] siblings= null; if (sibling != null) { siblings= new IJavaElement[] { sibling }; } String[] renamings= null; if (rename != null) { renamings= new String[] { rename }; } getJavaModel().copy(elements, containers, siblings, renamings, force, monitor); } /** * @see ISourceManipulation */ public void delete(boolean force, IProgressMonitor monitor) throws JavaModelException { IJavaElement[] elements= new IJavaElement[] { this }; getJavaModel().delete(elements, force, monitor); } public boolean equals(Object o) { if (!(o instanceof SourceRefElement)) return false; return this.occurrenceCount == ((SourceRefElement)o).occurrenceCount && super.equals(o); } /** * Returns the <code>ASTNode</code> that corresponds to this <code>JavaElement</code> or * <code>null</code> if there is no corresponding node. */ public ASTNode findNode(CompilationUnit ast) { DOMFinder finder= new DOMFinder(ast, this, false); try { return finder.search(); } catch (JavaModelException e) { // receiver doesn't exist return null; } } /* * @see JavaElement#generateInfos */ protected void generateInfos(Object info, HashMap newElements, IProgressMonitor pm) throws JavaModelException { Openable openableParent= (Openable)getOpenableParent(); if (openableParent == null) return; JavaElementInfo openableParentInfo= (JavaElementInfo)JavaModelManager.getJavaModelManager().getInfo(openableParent); if (openableParentInfo == null) { openableParent.generateInfos(openableParent.createElementInfo(), newElements, pm); } } public IAnnotation getAnnotation(String name) { return new Annotation(this, name); } public IAnnotation[] getAnnotations() throws JavaModelException { AnnotatableInfo info= (AnnotatableInfo)getElementInfo(); return info.annotations; } /** * @see IMember */ public ICompilationUnit getCompilationUnit() { return (ICompilationUnit)getAncestor(COMPILATION_UNIT); } /** * Elements within compilation units and class files have no corresponding resource. * * @see IJavaElement */ public IResource getCorrespondingResource() throws JavaModelException { if (!exists()) throw newNotPresentException(); return null; } /* * @see JavaElement */ public IJavaElement getHandleFromMemento(String token, MementoTokenizer memento, WorkingCopyOwner workingCopyOwner) { switch (token.charAt(0)) { case JEM_COUNT: return getHandleUpdatingCountFromMemento(memento, workingCopyOwner); } return this; } protected void getHandleMemento(StringBuffer buff) { super.getHandleMemento(buff); if (this.occurrenceCount > 1) { buff.append(JEM_COUNT); buff.append(this.occurrenceCount); } } /* * Update the occurence count of the receiver and creates a Java element handle from the given memento. * The given working copy owner is used only for compilation unit handles. */ public IJavaElement getHandleUpdatingCountFromMemento(MementoTokenizer memento, WorkingCopyOwner owner) { if (!memento.hasMoreTokens()) return this; this.occurrenceCount= Integer.parseInt(memento.nextToken()); if (!memento.hasMoreTokens()) return this; String token= memento.nextToken(); return getHandleFromMemento(token, memento, owner); } /* * @see IMember#getOccurrenceCount() */ public int getOccurrenceCount() { return this.occurrenceCount; } /** * Return the first instance of IOpenable in the hierarchy of this type (going up the hierarchy * from this type); */ public IOpenable getOpenableParent() { IJavaElement current= getParent(); while (current != null) { if (current instanceof IOpenable) { return (IOpenable)current; } current= current.getParent(); } return null; } /* * @see IJavaElement */ public IPath getPath() { return getParent().getPath(); } /* * @see IJavaElement */ public IResource resource() { return this.parent.resource(); } /** * @see ISourceReference */ public String getSource() throws JavaModelException { IOpenable openable= getOpenableParent(); IBuffer buffer= openable.getBuffer(); if (buffer == null) { return null; } ISourceRange range= getSourceRange(); int offset= range.getOffset(); int length= range.getLength(); if (offset == -1 || length == 0) { return null; } try { return buffer.getText(offset, length); } catch (RuntimeException e) { return null; } } /** * @see ISourceReference */ public ISourceRange getSourceRange() throws JavaModelException { SourceRefElementInfo info= (SourceRefElementInfo)getElementInfo(); return info.getSourceRange(); } /** * @see IJavaElement */ public IResource getUnderlyingResource() throws JavaModelException { if (!exists()) throw newNotPresentException(); return getParent().getUnderlyingResource(); } /** * @see IParent */ public boolean hasChildren() throws JavaModelException { return getChildren().length > 0; } /** * @see IJavaElement */ public boolean isStructureKnown() throws JavaModelException { // structure is always known inside an openable return true; } /** * @see ISourceManipulation */ public void move(IJavaElement container, IJavaElement sibling, String rename, boolean force, IProgressMonitor monitor) throws JavaModelException { if (container == null) { throw new IllegalArgumentException(Messages.operation_nullContainer); } IJavaElement[] elements= new IJavaElement[] { this }; IJavaElement[] containers= new IJavaElement[] { container }; IJavaElement[] siblings= null; if (sibling != null) { siblings= new IJavaElement[] { sibling }; } String[] renamings= null; if (rename != null) { renamings= new String[] { rename }; } getJavaModel().move(elements, containers, siblings, renamings, force, monitor); } /** * @see ISourceManipulation */ public void rename(String newName, boolean force, IProgressMonitor monitor) throws JavaModelException { if (newName == null) { throw new IllegalArgumentException(Messages.element_nullName); } IJavaElement[] elements= new IJavaElement[] { this }; IJavaElement[] dests= new IJavaElement[] { getParent() }; String[] renamings= new String[] { newName }; getJavaModel().rename(elements, dests, renamings, force, monitor); } protected void toStringName(StringBuffer buffer) { super.toStringName(buffer); if (this.occurrenceCount > 1) { buffer.append("#"); //$NON-NLS-1$ buffer.append(this.occurrenceCount); } } }