blob: b5ed04173a9632b448e55478b4c6d55f85c43472 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2015 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.search2.internal.ui.text;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.core.resources.IFile;
import org.eclipse.core.filebuffers.FileBuffers;
import org.eclipse.core.filebuffers.IFileBuffer;
import org.eclipse.core.filebuffers.IFileBufferListener;
import org.eclipse.core.filebuffers.ITextFileBuffer;
import org.eclipse.core.filebuffers.LocationKind;
import org.eclipse.jface.text.BadLocationException;
import org.eclipse.jface.text.IDocument;
import org.eclipse.jface.text.Position;
import org.eclipse.search.internal.ui.SearchPlugin;
import org.eclipse.search.ui.IQueryListener;
import org.eclipse.search.ui.ISearchQuery;
import org.eclipse.search.ui.ISearchResult;
import org.eclipse.search.ui.ISearchResultListener;
import org.eclipse.search.ui.NewSearchUI;
import org.eclipse.search.ui.SearchResultEvent;
import org.eclipse.search.ui.text.AbstractTextSearchResult;
import org.eclipse.search.ui.text.IFileMatchAdapter;
import org.eclipse.search.ui.text.Match;
import org.eclipse.search.ui.text.MatchEvent;
import org.eclipse.search.ui.text.RemoveAllEvent;
public class PositionTracker implements IQueryListener, ISearchResultListener, IFileBufferListener {
private Map<Match, Position> fMatchesToPositions= new HashMap<>();
private Map<Match, AbstractTextSearchResult> fMatchesToSearchResults= new HashMap<>();
private Map<ITextFileBuffer, Set<Match>> fFileBuffersToMatches= new HashMap<>();
private interface IFileBufferMatchOperation {
void run(ITextFileBuffer buffer, Match match);
}
public PositionTracker() {
NewSearchUI.addQueryListener(this);
FileBuffers.getTextFileBufferManager().addFileBufferListener(this);
}
// tracking search results --------------------------------------------------------------
@Override
public void queryAdded(ISearchQuery query) {
if (query.getSearchResult() instanceof AbstractTextSearchResult) {
query.getSearchResult().addListener(this);
}
}
@Override
public void queryRemoved(ISearchQuery query) {
ISearchResult result= query.getSearchResult();
if (result instanceof AbstractTextSearchResult) {
untrackAll((AbstractTextSearchResult)result);
result.removeListener(this);
}
}
// tracking matches ---------------------------------------------------------------------
@Override
public void searchResultChanged(SearchResultEvent e) {
if (e instanceof MatchEvent) {
MatchEvent evt= (MatchEvent)e;
Match[] matches = evt.getMatches();
int kind = evt.getKind();
AbstractTextSearchResult result = (AbstractTextSearchResult) e.getSearchResult();
for (int i = 0; i < matches.length; i++) {
ITextFileBuffer fb= getTrackedFileBuffer(result, matches[i].getElement());
if (fb != null) {
updateMatch(matches[i], fb, kind, result);
}
}
} else if (e instanceof RemoveAllEvent) {
RemoveAllEvent evt= (RemoveAllEvent)e;
ISearchResult result= evt.getSearchResult();
untrackAll((AbstractTextSearchResult)result);
}
}
private void updateMatch(Match match, ITextFileBuffer fb, int kind, AbstractTextSearchResult result) {
if (kind == MatchEvent.ADDED) {
trackPosition(result, fb, match);
} else if (kind == MatchEvent.REMOVED) {
untrackPosition(fb, match);
}
}
private void untrackAll(AbstractTextSearchResult result) {
Set<Match> matchSet= new HashSet<>(fMatchesToPositions.keySet());
for (Iterator<Match> matches= matchSet.iterator(); matches.hasNext();) {
Match match= matches.next();
AbstractTextSearchResult matchContainer= fMatchesToSearchResults.get(match);
if (result.equals(matchContainer)) {
ITextFileBuffer fb= getTrackedFileBuffer(result, match.getElement());
if (fb != null) {
untrackPosition(fb, match);
}
}
}
}
private void untrackPosition(ITextFileBuffer fb, Match match) {
Position position= fMatchesToPositions.get(match);
if (position != null) {
removeFileBufferMapping(fb, match);
fMatchesToSearchResults.remove(match);
fMatchesToPositions.remove(match);
fb.getDocument().removePosition(position);
}
}
private void trackPosition(AbstractTextSearchResult result, ITextFileBuffer fb, Match match) {
int offset = match.getOffset();
int length = match.getLength();
if (offset < 0 || length < 0)
return;
try {
IDocument doc= fb.getDocument();
Position position= new Position(offset, length);
if (match.getBaseUnit() == Match.UNIT_LINE) {
position= convertToCharacterPosition(position, doc);
}
doc.addPosition(position);
fMatchesToSearchResults.put(match, result);
fMatchesToPositions.put(match, position);
addFileBufferMapping(fb, match);
} catch (BadLocationException e) {
// the match is outside the document
result.removeMatch(match);
}
}
public static Position convertToCharacterPosition(Position linePosition, IDocument doc) throws BadLocationException {
int lineOffset= linePosition.getOffset();
int lineLength= linePosition.getLength();
int charOffset= doc.getLineOffset(lineOffset);
int charLength= 0;
if (lineLength > 0) {
int lastLine= lineOffset+lineLength-1;
int endPosition= doc.getLineOffset(lastLine)+doc.getLineLength(lastLine);
charLength= endPosition-charOffset;
}
return new Position(charOffset, charLength);
}
private void addFileBufferMapping(ITextFileBuffer fb, Match match) {
Set<Match> matches= fFileBuffersToMatches.get(fb);
if (matches == null) {
matches= new HashSet<>();
fFileBuffersToMatches.put(fb, matches);
}
matches.add(match);
}
private void removeFileBufferMapping(ITextFileBuffer fb, Match match) {
Set<Match> matches= fFileBuffersToMatches.get(fb);
if (matches != null) {
matches.remove(match);
if (matches.size() == 0)
fFileBuffersToMatches.remove(fb);
}
}
private ITextFileBuffer getTrackedFileBuffer(AbstractTextSearchResult result, Object element) {
IFileMatchAdapter adapter= result.getFileMatchAdapter();
if (adapter == null)
return null;
IFile file= adapter.getFile(element);
if (file == null)
return null;
if (!file.exists())
return null;
return FileBuffers.getTextFileBufferManager().getTextFileBuffer(file.getFullPath(), LocationKind.IFILE);
}
public Position getCurrentPosition(Match match) {
Position pos= fMatchesToPositions.get(match);
if (pos == null)
return pos;
AbstractTextSearchResult result= fMatchesToSearchResults.get(match);
if (match.getBaseUnit() == Match.UNIT_LINE && result != null) {
ITextFileBuffer fb= getTrackedFileBuffer(result, match.getElement());
if (fb != null) {
IDocument doc= fb.getDocument();
try {
pos= convertToLinePosition(pos, doc);
} catch (BadLocationException e) {
}
}
}
return pos;
}
public static Position convertToLinePosition(Position pos, IDocument doc) throws BadLocationException {
int offset= doc.getLineOfOffset(pos.getOffset());
int end= doc.getLineOfOffset(pos.getOffset()+pos.getLength());
int lineLength= end-offset;
if (pos.getLength() > 0 && lineLength == 0) {
// if the character length is > 0, add the last line, too
lineLength++;
}
return new Position(offset, lineLength);
}
public void dispose() {
NewSearchUI.removeQueryListener(this);
FileBuffers.getTextFileBufferManager().removeFileBufferListener(this);
}
// IFileBufferListener implementation ---------------------------------------------------------------------
@Override
public void bufferCreated(IFileBuffer buffer) {
final int[] trackCount= new int[1];
if (!(buffer instanceof ITextFileBuffer))
return;
IPath location= buffer.getLocation();
if (location == null)
return;
IFile file= FileBuffers.getWorkspaceFileAtLocation(location);
if (file == null)
return;
ISearchQuery[] queries= NewSearchUI.getQueries();
for (int i = 0; i < queries.length; i++) {
ISearchResult result = queries[i].getSearchResult();
if (result instanceof AbstractTextSearchResult) {
AbstractTextSearchResult textResult = (AbstractTextSearchResult) result;
IFileMatchAdapter adapter = textResult.getFileMatchAdapter();
if (adapter != null) {
Match[] matches = adapter.computeContainedMatches(textResult, file);
for (int j = 0; j < matches.length; j++) {
trackCount[0]++;
trackPosition((AbstractTextSearchResult) result, (ITextFileBuffer) buffer, matches[j]);
}
}
}
}
}
private void doForExistingMatchesIn(IFileBuffer buffer, IFileBufferMatchOperation operation) {
if (!(buffer instanceof ITextFileBuffer))
return;
Set<Match> matches= fFileBuffersToMatches.get(buffer);
if (matches != null) {
Set<Match> matchSet= new HashSet<>(matches);
for (Iterator<Match> matchIterator= matchSet.iterator(); matchIterator.hasNext();) {
Match element= matchIterator.next();
operation.run((ITextFileBuffer) buffer, element);
}
}
}
@Override
public void bufferDisposed(IFileBuffer buffer) {
final int[] trackCount= new int[1];
doForExistingMatchesIn(buffer, new IFileBufferMatchOperation() {
@Override
public void run(ITextFileBuffer textBuffer, Match match) {
trackCount[0]++;
untrackPosition(textBuffer, match);
}
});
}
@Override
public void bufferContentAboutToBeReplaced(IFileBuffer buffer) {
// not interesting for us.
}
@Override
public void bufferContentReplaced(IFileBuffer buffer) {
final int[] trackCount= new int[1];
doForExistingMatchesIn(buffer, new IFileBufferMatchOperation() {
@Override
public void run(ITextFileBuffer textBuffer, Match match) {
trackCount[0]++;
AbstractTextSearchResult result= fMatchesToSearchResults.get(match);
untrackPosition(textBuffer, match);
trackPosition(result, textBuffer, match);
}
});
}
@Override
public void stateChanging(IFileBuffer buffer) {
// not interesting for us
}
@Override
public void dirtyStateChanged(IFileBuffer buffer, boolean isDirty) {
if (isDirty)
return;
final int[] trackCount= new int[1];
doForExistingMatchesIn(buffer, new IFileBufferMatchOperation() {
@Override
public void run(ITextFileBuffer textBuffer, Match match) {
trackCount[0]++;
Position pos= fMatchesToPositions.get(match);
if (pos != null) {
if (pos.isDeleted()) {
AbstractTextSearchResult result= fMatchesToSearchResults.get(match);
// might be that the containing element has been removed.
if (result != null) {
result.removeMatch(match);
}
untrackPosition(textBuffer, match);
} else {
if (match.getBaseUnit() == Match.UNIT_LINE) {
try {
pos= convertToLinePosition(pos, textBuffer.getDocument());
} catch (BadLocationException e) {
SearchPlugin.getDefault().getLog().log(new Status(IStatus.ERROR, SearchPlugin.getID(), 0, e.getLocalizedMessage(), e));
}
}
match.setOffset(pos.getOffset());
match.setLength(pos.getLength());
}
}
}
});
}
@Override
public void stateValidationChanged(IFileBuffer buffer, boolean isStateValidated) {
// not interesting for us.
}
@Override
public void underlyingFileMoved(IFileBuffer buffer, IPath path) {
// not interesting for us.
}
@Override
public void underlyingFileDeleted(IFileBuffer buffer) {
// not interesting for us.
}
@Override
public void stateChangeFailed(IFileBuffer buffer) {
// not interesting for us.
}
@Override
public void queryStarting(ISearchQuery query) {
// not interested here
}
@Override
public void queryFinished(ISearchQuery query) {
// not interested
}
}