package com.redhat.ceylon.eclipse.code.editor;
import java.util.HashSet;
import java.util.Set;
import org.eclipse.core.resources.IMarker;
import org.eclipse.core.resources.IMarkerDelta;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.resources.IResourceChangeEvent;
import org.eclipse.core.resources.IResourceChangeListener;
import org.eclipse.core.resources.IResourceDelta;
import org.eclipse.core.resources.IResourceDeltaVisitor;
import org.eclipse.core.resources.ResourcesPlugin;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.ListenerList;
import org.eclipse.jface.text.source.AnnotationModelEvent;
import org.eclipse.jface.text.source.IAnnotationModel;
import org.eclipse.jface.text.source.IAnnotationModelListener;
import org.eclipse.jface.text.source.IAnnotationModelListenerExtension;
import org.eclipse.swt.widgets.Display;
/**
* Listens to resource deltas and filters for marker changes of type IMarker.PROBLEM.
* Viewers showing error ticks should register as listener to this type. Based on
* class of the same name from JDT/UI.
*
* @author Robert M. Fuhrer
*/
public class ProblemMarkerManager implements IResourceChangeListener,
IAnnotationModelListener, IAnnotationModelListenerExtension {
private ListenerList fListeners;
public ProblemMarkerManager() {
fListeners= new ListenerList();
}
/**
* Visitor used to determine whether the resource delta contains a marker change.
*/
private static class ProjectErrorVisitor implements IResourceDeltaVisitor {
private Set<IResource> fChangedElements;
public ProjectErrorVisitor(Set<IResource> changedElements) {
fChangedElements= changedElements;
}
public boolean visit(IResourceDelta delta) throws CoreException {
IResource res= delta.getResource();
if (res instanceof IProject && delta.getKind()==IResourceDelta.CHANGED) {
IProject project= (IProject) res;
if (!project.isAccessible()) {
// only track open Java projects
return false;
}
}
checkInvalidate(delta, res);
return true;
}
private void checkInvalidate(IResourceDelta delta, IResource resource) {
int kind= delta.getKind();
if (kind==IResourceDelta.REMOVED || kind==IResourceDelta.ADDED ||
(kind==IResourceDelta.CHANGED && isErrorDelta(delta))) {
// invalidate the resource and all parents
while (resource.getType()!=IResource.ROOT && fChangedElements.add(resource)) {
resource= resource.getParent();
}
}
}
private boolean isErrorDelta(IResourceDelta delta) {
if ((delta.getFlags() & IResourceDelta.MARKERS)!=0) {
IMarkerDelta[] markerDeltas= delta.getMarkerDeltas();
for(int i= 0; i < markerDeltas.length; i++) {
IMarkerDelta markerDelta= markerDeltas[i];
if (markerDelta.isSubtypeOf(IMarker.PROBLEM)) {
int kind= markerDelta.getKind();
if (kind==IResourceDelta.ADDED || kind==IResourceDelta.REMOVED)
return true;
int severity= markerDelta.getAttribute(IMarker.SEVERITY, -1);
int newSeverity= markerDelta.getMarker().getAttribute(IMarker.SEVERITY, -1);
if (newSeverity!=severity)
return true;
}
}
}
return false;
}
}
public void resourceChanged(IResourceChangeEvent event) {
Set<IResource> changedElements= new HashSet<IResource>();
try {
IResourceDelta delta= event.getDelta();
if (delta != null)
delta.accept(new ProjectErrorVisitor(changedElements));
}
catch (CoreException e) {
e.printStackTrace();
}
if (!changedElements.isEmpty()) {
IResource[] changes= (IResource[]) changedElements.toArray(new IResource[changedElements.size()]);
fireChanges(changes, true);
}
}
public void modelChanged(IAnnotationModel model) {}
public void modelChanged(AnnotationModelEvent event) {}
/**
* Adds a listener for problem marker changes.
*/
public void addListener(IProblemChangedListener listener) {
if (fListeners.isEmpty()) {
ResourcesPlugin.getWorkspace().addResourceChangeListener(this);
}
fListeners.add(listener);
}
/**
* Removes a <code>IProblemChangedListener</code>.
*/
public void removeListener(IProblemChangedListener listener) {
fListeners.remove(listener);
if (fListeners.isEmpty()) {
ResourcesPlugin.getWorkspace().removeResourceChangeListener(this);
}
}
private void fireChanges(final IResource[] changes, final boolean isMarkerChange) {
Display display = Display.getCurrent();
if (display==null)
display= Display.getDefault();
if (display!=null && !display.isDisposed()) {
display.asyncExec(new Runnable() {
public void run() {
Object[] listeners= fListeners.getListeners();
for(int i= 0; i < listeners.length; i++) {
IProblemChangedListener curr= (IProblemChangedListener) listeners[i];
curr.problemsChanged(changes, isMarkerChange);
}
}
});
}
}
}