1 /*******************************************************************************
2 * Copyright (c) 2000, 2003 IBM Corporation and others.
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Common Public License v1.0
5 * which accompanies this distribution, and is available at
6 * http://www.eclipse.org/legal/cpl-v10.html
9 * IBM Corporation - initial API and implementation
10 *******************************************************************************/
11 package net.sourceforge.phpdt.internal.ui.text.folding;
13 import java.util.ArrayList;
14 import java.util.HashMap;
15 import java.util.Iterator;
16 import java.util.List;
19 import net.sourceforge.phpdt.core.ElementChangedEvent;
20 import net.sourceforge.phpdt.core.IElementChangedListener;
21 import net.sourceforge.phpdt.core.IJavaElement;
22 import net.sourceforge.phpdt.core.IJavaElementDelta;
23 import net.sourceforge.phpdt.core.IParent;
24 import net.sourceforge.phpdt.core.ISourceRange;
25 import net.sourceforge.phpdt.core.ISourceReference;
26 import net.sourceforge.phpdt.core.IType;
27 import net.sourceforge.phpdt.core.JavaCore;
28 import net.sourceforge.phpdt.core.JavaModelException;
29 import net.sourceforge.phpdt.core.ToolFactory;
30 import net.sourceforge.phpdt.core.compiler.IScanner;
31 import net.sourceforge.phpdt.core.compiler.ITerminalSymbols;
32 import net.sourceforge.phpdt.core.compiler.InvalidInputException;
33 import net.sourceforge.phpdt.ui.IWorkingCopyManager;
34 import net.sourceforge.phpdt.ui.PreferenceConstants;
35 import net.sourceforge.phpdt.ui.text.folding.IJavaFoldingStructureProvider;
36 import net.sourceforge.phpeclipse.PHPeclipsePlugin;
37 import net.sourceforge.phpeclipse.phpeditor.PHPEditor;
38 import net.sourceforge.phpeclipse.phpeditor.PHPUnitEditor;
40 import org.eclipse.jface.preference.IPreferenceStore;
41 import org.eclipse.jface.text.BadLocationException;
42 import org.eclipse.jface.text.IDocument;
43 import org.eclipse.jface.text.IRegion;
44 import org.eclipse.jface.text.Position;
45 import org.eclipse.jface.text.Region;
46 import org.eclipse.jface.text.source.Annotation;
47 import org.eclipse.jface.text.source.IAnnotationModel;
48 import org.eclipse.jface.text.source.projection.IProjectionListener;
49 import org.eclipse.jface.text.source.projection.ProjectionAnnotation;
50 import org.eclipse.jface.text.source.projection.ProjectionAnnotationModel;
51 import org.eclipse.jface.text.source.projection.ProjectionViewer;
52 import org.eclipse.ui.texteditor.IDocumentProvider;
53 import org.eclipse.ui.texteditor.ITextEditor;
57 * Updates the projection model of a class file or compilation unit.
61 public class DefaultJavaFoldingStructureProvider implements IProjectionListener, IJavaFoldingStructureProvider {
63 private static class JavaProjectionAnnotation extends ProjectionAnnotation {
65 private IJavaElement fJavaElement;
66 private boolean fIsComment;
68 public JavaProjectionAnnotation(IJavaElement element, boolean isCollapsed, boolean isComment) {
70 fJavaElement= element;
71 fIsComment= isComment;
74 public IJavaElement getElement() {
78 public void setElement(IJavaElement element) {
79 fJavaElement= element;
82 public boolean isComment() {
86 public void setIsComment(boolean isComment) {
87 fIsComment= isComment;
91 private class ElementChangedListener implements IElementChangedListener {
94 * @see net.sourceforge.phpdt.core.IElementChangedListener#elementChanged(net.sourceforge.phpdt.core.ElementChangedEvent)
96 public void elementChanged(ElementChangedEvent e) {
97 IJavaElementDelta delta= findElement(fInput, e.getDelta());
102 private IJavaElementDelta findElement(IJavaElement target, IJavaElementDelta delta) {
104 if (delta == null || target == null)
107 IJavaElement element= delta.getElement();
109 if (element.getElementType() > IJavaElement.CLASS_FILE)
112 if (target.equals(element))
115 IJavaElementDelta[] children= delta.getAffectedChildren();
116 if (children == null || children.length == 0)
119 for (int i= 0; i < children.length; i++) {
120 IJavaElementDelta d= findElement(target, children[i]);
130 private IDocument fCachedDocument;
132 private ITextEditor fEditor;
133 private ProjectionViewer fViewer;
134 private IJavaElement fInput;
135 private IElementChangedListener fElementListener;
137 private boolean fAllowCollapsing= false;
138 private boolean fCollapseJavadoc= false;
139 private boolean fCollapseImportContainer= true;
140 private boolean fCollapseInnerTypes= true;
141 private boolean fCollapseMethods= false;
143 public DefaultJavaFoldingStructureProvider() {
146 public void install(ITextEditor editor, ProjectionViewer viewer) {
147 if (editor instanceof PHPEditor) {
150 fViewer.addProjectionListener(this);
154 public void uninstall() {
156 projectionDisabled();
157 fViewer.removeProjectionListener(this);
163 protected boolean isInstalled() {
164 return fEditor != null;
168 * @see org.eclipse.jface.text.source.projection.IProjectionListener#projectionEnabled()
170 public void projectionEnabled() {
171 if (fEditor instanceof PHPEditor) {
173 fElementListener= new ElementChangedListener();
174 JavaCore.addElementChangedListener(fElementListener);
179 * @see org.eclipse.jface.text.source.projection.IProjectionListener#projectionDisabled()
181 public void projectionDisabled() {
182 fCachedDocument= null;
183 if (fElementListener != null) {
184 JavaCore.removeElementChangedListener(fElementListener);
185 fElementListener= null;
189 public void initialize() {
194 initializePreferences();
198 IDocumentProvider provider= fEditor.getDocumentProvider();
199 fCachedDocument= provider.getDocument(fEditor.getEditorInput());
200 fAllowCollapsing= true;
202 if (fEditor instanceof PHPUnitEditor) {
203 IWorkingCopyManager manager= PHPeclipsePlugin.getDefault().getWorkingCopyManager();
204 fInput= manager.getWorkingCopy(fEditor.getEditorInput());
206 // else if (fEditor instanceof ClassFileEditor) {
207 // IClassFileEditorInput editorInput= (IClassFileEditorInput) fEditor.getEditorInput();
208 // fInput= editorInput.getClassFile();
211 if (fInput != null) {
212 ProjectionAnnotationModel model= (ProjectionAnnotationModel) fEditor.getAdapter(ProjectionAnnotationModel.class);
214 Map additions= computeAdditions((IParent) fInput);
215 model.removeAllAnnotations();
216 model.replaceAnnotations(null, additions);
221 fCachedDocument= null;
222 fAllowCollapsing= false;
226 private void initializePreferences() {
227 IPreferenceStore store= PHPeclipsePlugin.getDefault().getPreferenceStore();
228 fCollapseInnerTypes= store.getBoolean(PreferenceConstants.EDITOR_FOLDING_INNERTYPES);
229 fCollapseImportContainer= store.getBoolean(PreferenceConstants.EDITOR_FOLDING_IMPORTS);
230 fCollapseJavadoc= store.getBoolean(PreferenceConstants.EDITOR_FOLDING_JAVADOC);
231 fCollapseMethods= store.getBoolean(PreferenceConstants.EDITOR_FOLDING_METHODS);
234 private Map computeAdditions(IParent parent) {
235 Map map= new HashMap();
237 computeAdditions(parent.getChildren(), map);
238 } catch (JavaModelException x) {
243 private void computeAdditions(IJavaElement[] elements, Map map) throws JavaModelException {
244 for (int i= 0; i < elements.length; i++) {
245 IJavaElement element= elements[i];
247 computeAdditions(element, map);
249 if (element instanceof IParent) {
250 IParent parent= (IParent) element;
251 computeAdditions(parent.getChildren(), map);
256 private void computeAdditions(IJavaElement element, Map map) {
258 boolean createProjection= false;
260 boolean collapse= false;
261 switch (element.getElementType()) {
263 case IJavaElement.IMPORT_CONTAINER:
264 collapse= fAllowCollapsing && fCollapseImportContainer;
265 createProjection= true;
267 case IJavaElement.TYPE:
268 collapse= fAllowCollapsing && fCollapseInnerTypes && isInnerType((IType) element);
269 createProjection= true;
271 case IJavaElement.METHOD:
272 collapse= fAllowCollapsing && fCollapseMethods;
273 createProjection= true;
277 if (createProjection) {
278 IRegion[] regions= computeProjectionRanges(element);
279 if (regions != null) {
281 for (int i= 0; i < regions.length - 1; i++) {
282 Position position= createProjectionPosition(regions[i]);
283 if (position != null)
284 map.put(new JavaProjectionAnnotation(element, fAllowCollapsing && fCollapseJavadoc, true), position);
287 Position position= createProjectionPosition(regions[regions.length - 1]);
288 if (position != null)
289 map.put(new JavaProjectionAnnotation(element, collapse, false), position);
294 private boolean isInnerType(IType type) {
297 return type.isMember();
298 } catch (JavaModelException x) {
299 IJavaElement parent= type.getParent();
300 if (parent != null) {
301 int parentType= parent.getElementType();
302 return (parentType != IJavaElement.COMPILATION_UNIT && parentType != IJavaElement.CLASS_FILE);
309 private IRegion[] computeProjectionRanges(IJavaElement element) {
312 if (element instanceof ISourceReference) {
313 ISourceReference reference= (ISourceReference) element;
314 ISourceRange range= reference.getSourceRange();
315 String contents= reference.getSource();
316 if (contents == null)
319 IScanner scanner= ToolFactory.createScanner(true/*tokenizeComments*/, false, false, true /* phpMode */ );
320 scanner.setSource(contents.toCharArray());
322 List regions= new ArrayList();
323 int shift= range.getOffset();
327 int token= scanner.getNextToken();
328 start= shift + scanner.getCurrentTokenStartPosition();
331 case ITerminalSymbols.TokenNameCOMMENT_PHPDOC: // COMMENT_JAVADOC:
332 case ITerminalSymbols.TokenNameCOMMENT_BLOCK: {
333 int end= shift + scanner.getCurrentTokenEndPosition() + 1;
334 regions.add(new Region(start, end - start));
336 case ITerminalSymbols.TokenNameCOMMENT_LINE:
343 regions.add(new Region(start, range.getOffset() + range.getLength() - start));
345 if (regions.size() > 0) {
346 IRegion[] result= new IRegion[regions.size()];
347 regions.toArray(result);
351 } catch (JavaModelException e) {
352 } catch (InvalidInputException e) {
358 private Position createProjectionPosition(IRegion region) {
360 if (fCachedDocument == null)
365 int start= fCachedDocument.getLineOfOffset(region.getOffset());
366 int end= fCachedDocument.getLineOfOffset(region.getOffset() + region.getLength());
368 int offset= fCachedDocument.getLineOffset(start);
369 int endOffset= fCachedDocument.getLineOffset(end + 1);
370 return new Position(offset, endOffset - offset);
373 } catch (BadLocationException x) {
379 protected void processDelta(IJavaElementDelta delta) {
384 ProjectionAnnotationModel model= (ProjectionAnnotationModel) fEditor.getAdapter(ProjectionAnnotationModel.class);
390 IDocumentProvider provider= fEditor.getDocumentProvider();
391 fCachedDocument= provider.getDocument(fEditor.getEditorInput());
392 fAllowCollapsing= false;
394 Map additions= new HashMap();
395 List deletions= new ArrayList();
396 List updates= new ArrayList();
398 Map updated= computeAdditions((IParent) fInput);
399 Map previous= createAnnotationMap(model);
402 Iterator e= updated.keySet().iterator();
403 while (e.hasNext()) {
404 JavaProjectionAnnotation annotation= (JavaProjectionAnnotation) e.next();
405 IJavaElement element= annotation.getElement();
406 Position position= (Position) updated.get(annotation);
408 List annotations= (List) previous.get(element);
409 if (annotations == null) {
411 additions.put(annotation, position);
415 Iterator x= annotations.iterator();
416 while (x.hasNext()) {
417 JavaProjectionAnnotation a= (JavaProjectionAnnotation) x.next();
418 if (annotation.isComment() == a.isComment()) {
419 Position p= model.getPosition(a);
420 if (p != null && !position.equals(p)) {
421 p.setOffset(position.getOffset());
422 p.setLength(position.getLength());
430 if (annotations.isEmpty())
431 previous.remove(element);
435 e= previous.values().iterator();
436 while (e.hasNext()) {
437 List list= (List) e.next();
438 int size= list.size();
439 for (int i= 0; i < size; i++)
440 deletions.add(list.get(i));
443 match(model, deletions, additions, updates);
445 Annotation[] removals= new Annotation[deletions.size()];
446 deletions.toArray(removals);
447 Annotation[] changes= new Annotation[updates.size()];
448 updates.toArray(changes);
449 model.modifyAnnotations(removals, additions, changes);
452 fCachedDocument= null;
453 fAllowCollapsing= true;
457 private void match(ProjectionAnnotationModel model, List deletions, Map additions, List changes) {
458 if (deletions.isEmpty() || (additions.isEmpty() && changes.isEmpty()))
461 List newDeletions= new ArrayList();
462 List newChanges= new ArrayList();
464 Iterator deletionIterator= deletions.iterator();
465 outer: while (deletionIterator.hasNext()) {
466 JavaProjectionAnnotation deleted= (JavaProjectionAnnotation) deletionIterator.next();
467 Position deletedPosition= model.getPosition(deleted);
468 if (deletedPosition == null)
471 Iterator changesIterator= changes.iterator();
472 while (changesIterator.hasNext()) {
473 JavaProjectionAnnotation changed= (JavaProjectionAnnotation) changesIterator.next();
474 if (deleted.isComment() == changed.isComment()) {
475 Position changedPosition= model.getPosition(changed);
476 if (changedPosition == null)
479 if (deletedPosition.getOffset() == changedPosition.getOffset()) {
481 deletedPosition.setLength(changedPosition.getLength());
482 deleted.setElement(changed.getElement());
484 deletionIterator.remove();
485 newChanges.add(deleted);
487 changesIterator.remove();
488 newDeletions.add(changed);
495 Iterator additionsIterator= additions.keySet().iterator();
496 while (additionsIterator.hasNext()) {
497 JavaProjectionAnnotation added= (JavaProjectionAnnotation) additionsIterator.next();
498 if (deleted.isComment() == added.isComment()) {
499 Position addedPosition= (Position) additions.get(added);
501 if (deletedPosition.getOffset() == addedPosition.getOffset()) {
503 deletedPosition.setLength(addedPosition.getLength());
504 deleted.setElement(added.getElement());
506 deletionIterator.remove();
507 newChanges.add(deleted);
509 additionsIterator.remove();
517 deletions.addAll(newDeletions);
518 changes.addAll(newChanges);
521 private Map createAnnotationMap(IAnnotationModel model) {
522 Map map= new HashMap();
523 Iterator e= model.getAnnotationIterator();
524 while (e.hasNext()) {
525 Object annotation= e.next();
526 if (annotation instanceof JavaProjectionAnnotation) {
527 JavaProjectionAnnotation java= (JavaProjectionAnnotation) annotation;
528 List list= (List) map.get(java.getElement());
530 list= new ArrayList(2);
531 map.put(java.getElement(), list);