/******************************************************************************* * Copyright (c) 2000, 2006 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.erlide.ui.util; 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.erlide.util.ErlLogger; /** * Listens to resource deltas and filters for marker changes of type * IMarker.PROBLEM Viewers showing error ticks should register as listener to * this type. */ public class ProblemMarkerManager implements IResourceChangeListener, IAnnotationModelListener, IAnnotationModelListenerExtension { /** * Visitors used to look if the element change delta contains a marker * change. */ private static class ProjectErrorVisitor implements IResourceDeltaVisitor { private final Set<IResource> fChangedElements; public ProjectErrorVisitor(final Set<IResource> changedElements) { fChangedElements = changedElements; } @Override public boolean visit(final IResourceDelta delta) throws CoreException { final IResource res = delta.getResource(); if (res instanceof IProject && delta.getKind() == IResourceDelta.CHANGED) { final IProject project = (IProject) res; if (!project.isAccessible()) { // only track open Erlang projects return false; } } checkInvalidate(delta, res); return true; } private void checkInvalidate(final IResourceDelta delta, final IResource resource0) { IResource resource = resource0; final 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(final IResourceDelta delta) { if ((delta.getFlags() & IResourceDelta.MARKERS) != 0) { final IMarkerDelta[] markerDeltas = delta.getMarkerDeltas(); for (int i = 0; i < markerDeltas.length; i++) { if (markerDeltas[i].isSubtypeOf(IMarker.PROBLEM)) { final int kind = markerDeltas[i].getKind(); if (kind == IResourceDelta.ADDED || kind == IResourceDelta.REMOVED) { return true; } final int severity = markerDeltas[i] .getAttribute(IMarker.SEVERITY, -1); final int newSeverity = markerDeltas[i].getMarker() .getAttribute(IMarker.SEVERITY, -1); if (newSeverity != severity) { return true; } } } } return false; } } private final ListenerList fListeners; public ProblemMarkerManager() { fListeners = new ListenerList(); } /* * @see IResourceChangeListener#resourceChanged */ @Override public void resourceChanged(final IResourceChangeEvent event) { final HashSet<IResource> changedElements = new HashSet<>(); try { final IResourceDelta delta = event.getDelta(); if (delta != null) { delta.accept(new ProjectErrorVisitor(changedElements)); } } catch (final CoreException e) { ErlLogger.error(e); } if (!changedElements.isEmpty()) { final IResource[] changes = changedElements .toArray(new IResource[changedElements.size()]); fireChanges(changes, true); } } @Override public void modelChanged(final IAnnotationModel model) { // no action } @Override public void modelChanged(final AnnotationModelEvent event) { if (event instanceof ErlangModuleAnnotationModelEvent) { final ErlangModuleAnnotationModelEvent emEvent = (ErlangModuleAnnotationModelEvent) event; if (emEvent.includesProblemMarkerAnnotationChanges()) { final IResource[] changes = new IResource[] { emEvent.getUnderlyingResource() }; fireChanges(changes, false); } } } /** * Adds a listener for problem marker changes. */ public void addListener(final IProblemChangedListener listener) { if (fListeners.isEmpty()) { ResourcesPlugin.getWorkspace().addResourceChangeListener(this); // JavaPlugin.getDefault().getCompilationUnitDocumentProvider() // .addGlobalAnnotationModelListener(this); // FIXME use some kind of Erlide model-listener } fListeners.add(listener); } /** * Removes a <code>IProblemChangedListener</code>. */ public void removeListener(final IProblemChangedListener listener) { fListeners.remove(listener); if (fListeners.isEmpty()) { ResourcesPlugin.getWorkspace().removeResourceChangeListener(this); // JavaPlugin.getDefault().getCompilationUnitDocumentProvider() // .removeGlobalAnnotationModelListener(this); // FIXME use some kind of Erlide model-listener } } private void fireChanges(final IResource[] changes, final boolean isMarkerChange) { DisplayUtils.asyncExec(new Runnable() { @Override @SuppressWarnings("synthetic-access") public void run() { final Object[] listeners = fListeners.getListeners(); for (int i = 0; i < listeners.length; i++) { final IProblemChangedListener curr = (IProblemChangedListener) listeners[i]; curr.problemsChanged(changes, isMarkerChange); } } }); } }