blob: 6e3102316505361be3a0a082c467f8c4f91d4094 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2003 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Common Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/cpl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.search.internal.ui;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.Map;
import org.eclipse.core.resources.IMarker;
import org.eclipse.core.resources.IMarkerDelta;
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.runtime.CoreException;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.jface.dialogs.MessageDialog;
import org.eclipse.jface.dialogs.ProgressMonitorDialog;
import org.eclipse.jface.util.Assert;
import org.eclipse.jface.viewers.Viewer;
import org.eclipse.ui.actions.WorkspaceModifyOperation;
import org.eclipse.search.ui.IGroupByKeyComputer;
import org.eclipse.search.ui.SearchUI;
import org.eclipse.search.internal.ui.util.ExceptionHandler;
/**
* Manage search results
*/
public class SearchManager implements IResourceChangeListener {
static final SearchManager fgDefault= new SearchManager();
Search fCurrentSearch= null;
private SearchManager() {
SearchPlugin.getWorkspace().addResourceChangeListener(this);
}
private HashSet fListeners= new HashSet();
private LinkedList fPreviousSearches= new LinkedList();
private boolean fIsRemoveAll= false;
public static SearchManager getDefault() {
return fgDefault;
}
/**
* Returns the list with previous searches (ISearch).
*/
LinkedList getPreviousSearches() {
return fPreviousSearches;
}
/**
* Returns the list with current (last) results
*/
ArrayList getCurrentResults() {
if (fCurrentSearch == null)
return new ArrayList(0);
else
return (ArrayList)fCurrentSearch.getResults();
}
public Search getCurrentSearch() {
return fCurrentSearch;
}
void removeAllSearches() {
SearchPlugin.getWorkspace().removeResourceChangeListener(this);
WorkspaceModifyOperation op= new WorkspaceModifyOperation(null) {
protected void execute(IProgressMonitor monitor) throws CoreException {
monitor.beginTask(SearchMessages.getString("SearchManager.updating"), 100); //$NON-NLS-1$
SearchPlugin.getWorkspace().getRoot().deleteMarkers(SearchUI.SEARCH_MARKER, true, IResource.DEPTH_INFINITE);
monitor.worked(100);
monitor.done();
}
};
boolean isAutoBuilding= SearchPlugin.getWorkspace().isAutoBuilding();
if (isAutoBuilding)
// disable auto-build during search operation
SearchPlugin.setAutoBuilding(false);
try {
ProgressMonitorDialog dialog= new ProgressMonitorDialog(getShell());
dialog.run(true, true, op);
} catch (InvocationTargetException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.deleteMarkers.title"), SearchMessages.getString("Search.Error.deleteMarkers.message")); //$NON-NLS-2$ //$NON-NLS-1$
} catch (InterruptedException e) {
// Do nothing. Operation has been canceled.
} finally {
SearchPlugin.getWorkspace().addResourceChangeListener(this);
if (isAutoBuilding)
// enable auto-building again
SearchPlugin.setAutoBuilding(true);
}
// clear searches
fPreviousSearches= new LinkedList();
fCurrentSearch= null;
// update viewers
Iterator iter= fListeners.iterator();
while (iter.hasNext()) {
SearchResultViewer viewer= (SearchResultViewer)iter.next();
handleAllSearchesRemoved(viewer);
}
}
private void handleAllSearchesRemoved(SearchResultViewer viewer) {
viewer.handleAllSearchesRemoved();
}
void setCurrentSearch(final Search search) {
if (fCurrentSearch == search)
return;
SearchPlugin.getWorkspace().removeResourceChangeListener(this);
WorkspaceModifyOperation op= new WorkspaceModifyOperation(null) {
protected void execute(IProgressMonitor monitor) throws CoreException {
internalSetCurrentSearch(search, monitor);
}
};
boolean isAutoBuilding= SearchPlugin.getWorkspace().isAutoBuilding();
if (isAutoBuilding)
// disable auto-build during search operation
SearchPlugin.setAutoBuilding(false);
try {
ProgressMonitorDialog dialog= new ProgressMonitorDialog(getShell());
dialog.run(true, true, op);
} catch (InvocationTargetException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.switchSearch.title"), SearchMessages.getString("Search.Error.switchSearch.message")); //$NON-NLS-2$ //$NON-NLS-1$
} catch (InterruptedException e) {
// Do nothing. Operation has been canceled.
} finally {
SearchPlugin.getWorkspace().addResourceChangeListener(this);
if (isAutoBuilding)
// enable auto-building again
SearchPlugin.setAutoBuilding(true);
}
getPreviousSearches().remove(search);
getPreviousSearches().addFirst(search);
}
void internalSetCurrentSearch(final Search search, IProgressMonitor monitor) {
if (fCurrentSearch != null)
fCurrentSearch.backupMarkers();
final Search previousSearch= fCurrentSearch;
fCurrentSearch= search;
monitor.beginTask(SearchMessages.getString("SearchManager.updating"), getCurrentResults().size() + 20); //$NON-NLS-1$
// remove current search markers
try {
SearchPlugin.getWorkspace().getRoot().deleteMarkers(SearchUI.SEARCH_MARKER, true, IResource.DEPTH_INFINITE);
} catch (CoreException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.deleteMarkers.title"), SearchMessages.getString("Search.Error.deleteMarkers.message")); //$NON-NLS-2$ //$NON-NLS-1$
}
monitor.worked(10);
// add search markers
Iterator iter= getCurrentResults().iterator();
ArrayList emptyEntries= new ArrayList(10);
boolean filesChanged= false;
boolean filesDeleted= false;
IGroupByKeyComputer groupByKeyComputer= getCurrentSearch().getGroupByKeyComputer();
while (iter.hasNext()) {
monitor.worked(1);
SearchResultViewEntry entry= (SearchResultViewEntry)iter.next();
Iterator attrPerMarkerIter= entry.getAttributesPerMarker().iterator();
entry.clearMarkerList();
if (entry.getResource() == null || !entry.getResource().exists()) {
emptyEntries.add(entry);
filesDeleted= true;
continue;
}
while (attrPerMarkerIter.hasNext()) {
IMarker newMarker= null;
try {
newMarker= entry.getResource().createMarker(entry.getMarkerType());
} catch (CoreException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.createMarker.title"), SearchMessages.getString("Search.Error.createMarker.message")); //$NON-NLS-2$ //$NON-NLS-1$
continue;
}
try {
newMarker.setAttributes((Map)attrPerMarkerIter.next());
if (groupByKeyComputer !=null && groupByKeyComputer.computeGroupByKey(newMarker) == null) {
filesDeleted= true;
newMarker.delete();
continue;
}
} catch (CoreException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.markerAttributeAccess.title"), SearchMessages.getString("Search.Error.markerAttributeAccess.message")); //$NON-NLS-2$ //$NON-NLS-1$
}
entry.add(newMarker);
}
if (entry.getMatchCount() == 0)
emptyEntries.add(entry);
else if (!filesChanged && entry.getResource().getModificationStamp() != entry.getModificationStamp())
filesChanged= true;
}
getCurrentResults().removeAll(emptyEntries);
monitor.worked(10);
String warningMessage= null;
Display display= getDisplay();
if (filesChanged)
warningMessage= SearchMessages.getString("SearchManager.resourceChanged"); //$NON-NLS-1$
if (filesDeleted) {
if (warningMessage == null)
warningMessage= ""; //$NON-NLS-1$
else
warningMessage += "\n"; //$NON-NLS-1$
warningMessage += SearchMessages.getString("SearchManager.resourceDeleted"); //$NON-NLS-1$
}
if (warningMessage != null) {
if (display != null && !display.isDisposed()) {
final String warningTitle= SearchMessages.getString("SearchManager.resourceChangedWarning"); //$NON-NLS-1$
final String warningMsg= warningMessage;
display.syncExec(new Runnable() {
public void run() {
MessageDialog.openWarning(getShell(), warningTitle, warningMsg);
}
});
}
}
// update viewers
iter= fListeners.iterator();
if (display != null && !display.isDisposed()) {
final Viewer visibleViewer= ((SearchResultView)SearchPlugin.getSearchResultView()).getViewer();
while (iter.hasNext()) {
final SearchResultViewer viewer= (SearchResultViewer)iter.next();
display.syncExec(new Runnable() {
public void run() {
if (previousSearch != null && viewer == visibleViewer)
previousSearch.setSelection(viewer.getSelection());
viewer.setInput(null);
viewer.setPageId(search.getPageId());
viewer.setGotoMarkerAction(search.getGotoMarkerAction());
viewer.setContextMenuTarget(search.getContextMenuContributor());
viewer.setActionGroupFactory(null);
viewer.setInput(getCurrentResults());
viewer.setActionGroupFactory(search.getActionGroupFactory());
viewer.setSelection(fCurrentSearch.getSelection(), true);
}
});
}
}
monitor.done();
}
/**
* Returns the number of matches
*/
int getCurrentItemCount() {
if (fCurrentSearch != null)
return fCurrentSearch.getItemCount();
else
return 0;
}
void removeAllResults() {
fIsRemoveAll= true;
try {
SearchPlugin.getWorkspace().getRoot().deleteMarkers(SearchUI.SEARCH_MARKER, true, IResource.DEPTH_INFINITE);
} catch (CoreException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.deleteMarkers.title"), SearchMessages.getString("Search.Error.deleteMarkers.message")); //$NON-NLS-2$ //$NON-NLS-1$
fIsRemoveAll= false;
}
}
void addNewSearch(final Search newSearch) {
SearchPlugin.getWorkspace().removeResourceChangeListener(this);
// Clear the viewers
Iterator iter= fListeners.iterator();
Display display= getDisplay();
if (display != null && !display.isDisposed()) {
final Viewer visibleViewer= ((SearchResultView)SearchPlugin.getSearchResultView()).getViewer();
while (iter.hasNext()) {
final SearchResultViewer viewer= (SearchResultViewer)iter.next();
display.syncExec(new Runnable() {
public void run() {
if (fCurrentSearch != null && viewer == visibleViewer)
fCurrentSearch.setSelection(viewer.getSelection());
setNewSearch(viewer, newSearch);
}
});
}
}
if (fCurrentSearch != null) {
if (fCurrentSearch.isSameSearch(newSearch))
getPreviousSearches().remove(fCurrentSearch);
else
fCurrentSearch.backupMarkers();
}
fCurrentSearch= newSearch;
getPreviousSearches().addFirst(fCurrentSearch);
// Remove the markers
try {
SearchPlugin.getWorkspace().getRoot().deleteMarkers(SearchUI.SEARCH_MARKER, true, IResource.DEPTH_INFINITE);
} catch (CoreException ex) {
ExceptionHandler.handle(ex, SearchMessages.getString("Search.Error.deleteMarkers.title"), SearchMessages.getString("Search.Error.deleteMarkers.message")); //$NON-NLS-2$ //$NON-NLS-1$
}
}
void searchFinished(ArrayList results) {
Assert.isNotNull(results);
getCurrentSearch().setResults(results);
Display display= getDisplay();
if (display == null || display.isDisposed())
return;
if (Thread.currentThread() == display.getThread())
handleNewSearchResult();
else {
display.syncExec(new Runnable() {
public void run() {
handleNewSearchResult();
}
});
}
SearchPlugin.getWorkspace().addResourceChangeListener(this);
}
//--- Change event handling -------------------------------------------------
void addSearchChangeListener(SearchResultViewer viewer) {
fListeners.add(viewer);
}
void removeSearchChangeListener(SearchResultViewer viewer) {
Assert.isNotNull(viewer);
fListeners.remove(viewer);
}
private final void handleSearchMarkersChanged(IMarkerDelta[] markerDeltas) {
if (fIsRemoveAll) {
handleRemoveAll();
fIsRemoveAll= false;
return;
}
Iterator iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).getControl().setRedraw(false);
for (int i=0; i < markerDeltas.length; i++) {
handleSearchMarkerChanged(markerDeltas[i]);
}
iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).getControl().setRedraw(true);
}
private void handleSearchMarkerChanged(IMarkerDelta markerDelta) {
int kind= markerDelta.getKind();
// don't listen for adds will be done by ISearchResultView.addMatch(...)
if (((kind & IResourceDelta.REMOVED) != 0))
handleRemoveMatch(markerDelta.getMarker());
else if ((kind & IResourceDelta.CHANGED) != 0)
handleUpdateMatch(markerDelta.getMarker());
}
private void handleRemoveAll() {
if (fCurrentSearch != null)
fCurrentSearch.removeResults();
Iterator iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).handleRemoveAll();
}
private void handleNewSearchResult() {
Iterator iter= fListeners.iterator();
while (iter.hasNext()) {
SearchResultViewer viewer= (SearchResultViewer)iter.next();
viewer.setInput(getCurrentResults());
}
}
private void setNewSearch(SearchResultViewer viewer, Search search) {
viewer.setInput(null);
viewer.clearTitle();
viewer.setPageId(search.getPageId());
viewer.setGotoMarkerAction(search.getGotoMarkerAction());
viewer.setContextMenuTarget(search.getContextMenuContributor());
viewer.setActionGroupFactory(search.getActionGroupFactory());
}
private void handleRemoveMatch(IMarker marker) {
SearchResultViewEntry entry= findEntry(marker);
if (entry != null) {
entry.remove(marker);
if (entry.getMatchCount() == 0) {
getCurrentResults().remove(entry);
Iterator iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).handleRemoveMatch(entry);
}
else {
Iterator iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).handleUpdateMatch(entry, true);
}
}
}
private void handleUpdateMatch(IMarker marker) {
SearchResultViewEntry entry= findEntry(marker);
if (entry != null) {
Iterator iter= fListeners.iterator();
while (iter.hasNext())
((SearchResultViewer)iter.next()).handleUpdateMatch(entry, false);
}
}
private SearchResultViewEntry findEntry(IMarker marker) {
Iterator entries= getCurrentResults().iterator();
while (entries.hasNext()) {
SearchResultViewEntry entry= (SearchResultViewEntry)entries.next();
if (entry.contains(marker))
return entry;
}
return null;
}
/**
* Received a resource event. Since the delta could be created in a
* separate thread this methods post the event into the viewer's
* display thread.
*/
public final void resourceChanged(final IResourceChangeEvent event) {
if (event == null)
return;
final IMarkerDelta[] markerDeltas= event.findMarkerDeltas(SearchUI.SEARCH_MARKER, true);
if (markerDeltas == null || markerDeltas.length < 1)
return;
Display display= getDisplay();
if (display == null || display.isDisposed())
return;
Runnable runnable= new Runnable() {
public void run() {
if (getCurrentSearch() != null) {
handleSearchMarkersChanged(markerDeltas);
// update title and actions
Iterator iter= fListeners.iterator();
while (iter.hasNext()) {
SearchResultViewer viewer= (SearchResultViewer)iter.next();
viewer.enableActions();
viewer.updateTitle();
}
}
}
};
display.syncExec(runnable);
}
/**
* Find and return a valid display
*/
private Display getDisplay() {
Iterator iter= fListeners.iterator();
while (iter.hasNext()) {
Control control= ((Viewer)iter.next()).getControl();
if (control != null && !control.isDisposed()) {
Display display= control.getDisplay();
if (display != null && !display.isDisposed())
return display;
}
}
return null;
}
/**
* Find and return a valid shell
*/
private Shell getShell() {
return SearchPlugin.getActiveWorkbenchShell();
}
}