blob: ec44c230bd29c1d90cd12eec941b118bedf46bd0 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2002, 2004 GEBIT Gesellschaft fuer EDV-Beratung
* und Informatik-Technologien mbH,
* Berlin, Duesseldorf, Frankfurt (Germany) 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:
* GEBIT Gesellschaft fuer EDV-Beratung und Informatik-Technologien mbH - initial API and implementation
* IBM Corporation - bug fixes
* John-Mason P. Shackelford - bug 40255
*******************************************************************************/
package org.eclipse.ant.internal.ui.editor;
import java.util.ResourceBundle;
import org.eclipse.ant.internal.ui.editor.model.AntElementNode;
import org.eclipse.ant.internal.ui.editor.model.AntProjectNode;
import org.eclipse.ant.internal.ui.editor.outline.AntEditorContentOutlinePage;
import org.eclipse.ant.internal.ui.editor.outline.AntModel;
import org.eclipse.ant.internal.ui.editor.outline.XMLCore;
import org.eclipse.ant.internal.ui.editor.text.AnnotationAccess;
import org.eclipse.ant.internal.ui.editor.text.AntEditorDocumentProvider;
import org.eclipse.ant.internal.ui.editor.text.IAntEditorColorConstants;
import org.eclipse.ant.internal.ui.editor.text.IReconcilingParticipant;
import org.eclipse.ant.internal.ui.model.AntUIPlugin;
import org.eclipse.ant.internal.ui.model.AntUtil;
import org.eclipse.ant.internal.ui.model.IAntUIHelpContextIds;
import org.eclipse.ant.internal.ui.model.IAntUIPreferenceConstants;
import org.eclipse.ant.internal.ui.preferences.AntEditorPreferenceConstants;
import org.eclipse.core.resources.IFile;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.jdt.ui.actions.IJavaEditorActionDefinitionIds;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.action.IMenuManager;
import org.eclipse.jface.preference.IPreferenceStore;
import org.eclipse.jface.preference.PreferenceConverter;
import org.eclipse.jface.resource.JFaceResources;
import org.eclipse.jface.text.BadLocationException;
import org.eclipse.jface.text.DefaultLineTracker;
import org.eclipse.jface.text.DocumentCommand;
import org.eclipse.jface.text.IAutoIndentStrategy;
import org.eclipse.jface.text.IDocument;
import org.eclipse.jface.text.ILineTracker;
import org.eclipse.jface.text.IRegion;
import org.eclipse.jface.text.ITextSelection;
import org.eclipse.jface.text.source.IAnnotationAccess;
import org.eclipse.jface.text.source.IOverviewRuler;
import org.eclipse.jface.text.source.ISourceViewer;
import org.eclipse.jface.text.source.IVerticalRuler;
import org.eclipse.jface.text.source.SourceViewer;
import org.eclipse.jface.text.source.SourceViewerConfiguration;
import org.eclipse.jface.util.IPropertyChangeListener;
import org.eclipse.jface.util.PropertyChangeEvent;
import org.eclipse.jface.viewers.IPostSelectionProvider;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.jface.viewers.ISelectionChangedListener;
import org.eclipse.jface.viewers.ISelectionProvider;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.jface.viewers.SelectionChangedEvent;
import org.eclipse.swt.custom.StyledText;
import org.eclipse.swt.graphics.Image;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.ui.IEditorInput;
import org.eclipse.ui.IPartService;
import org.eclipse.ui.IWorkbenchPage;
import org.eclipse.ui.IWorkbenchPart;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.PartInitException;
import org.eclipse.ui.editors.text.EditorsUI;
import org.eclipse.ui.editors.text.TextEditor;
import org.eclipse.ui.ide.IDE;
import org.eclipse.ui.texteditor.AbstractDecoratedTextEditorPreferenceConstants;
import org.eclipse.ui.texteditor.ChainedPreferenceStore;
import org.eclipse.ui.texteditor.ContentAssistAction;
import org.eclipse.ui.texteditor.IDocumentProvider;
import org.eclipse.ui.texteditor.IEditorStatusLine;
import org.eclipse.ui.texteditor.ITextEditorActionDefinitionIds;
import org.eclipse.ui.texteditor.TextOperationAction;
import org.eclipse.ui.views.contentoutline.ContentOutline;
import org.eclipse.ui.views.contentoutline.IContentOutlinePage;
/**
* The actual editor implementation for Eclipse's Ant integration.
*/
public class AntEditor extends TextEditor implements IReconcilingParticipant {
//TODO the framework does not currently support/listen to the color registry
//https://bugs.eclipse.org/bugs/show_bug.cgi?id=54554
private IPropertyChangeListener fColorChangeListener= new IPropertyChangeListener() {
public void propertyChange(PropertyChangeEvent event) {
handlePreferenceStoreChanged(event);
}
};
/**
* Updates the Ant outline page selection and this editor's range indicator.
*
* @since 3.0
*/
private class EditorSelectionChangedListener implements ISelectionChangedListener {
/**
* Installs this selection changed listener with the given selection provider. If
* the selection provider is a post selection provider, post selection changed
* events are the preferred choice, otherwise normal selection changed events
* are requested.
*
* @param selectionProvider
*/
public void install(ISelectionProvider selectionProvider) {
if (selectionProvider == null) {
return;
}
if (selectionProvider instanceof IPostSelectionProvider) {
IPostSelectionProvider provider= (IPostSelectionProvider) selectionProvider;
provider.addPostSelectionChangedListener(this);
} else {
selectionProvider.addSelectionChangedListener(this);
}
}
/**
* Removes this selection changed listener from the given selection provider.
*
* @param selectionProviderstyle
*/
public void uninstall(ISelectionProvider selectionProvider) {
if (selectionProvider == null) {
return;
}
if (selectionProvider instanceof IPostSelectionProvider) {
IPostSelectionProvider provider= (IPostSelectionProvider) selectionProvider;
provider.removePostSelectionChangedListener(this);
} else {
selectionProvider.removeSelectionChangedListener(this);
}
}
/*
* @see org.eclipse.jface.viewers.ISelectionChangedListener#selectionChanged(org.eclipse.jface.viewers.SelectionChangedEvent)
*/
public void selectionChanged(SelectionChangedEvent event) {
AntModel model= getAntModel();
if (model == null) { //external file
return;
}
ISelection selection= event.getSelection();
AntElementNode node= null;
if (selection instanceof ITextSelection) {
ITextSelection textSelection= (ITextSelection)selection;
int offset= textSelection.getOffset();
node= model.getNode(offset, false);
}
if (AntUIPlugin.getDefault().getPreferenceStore().getBoolean(IAntUIPreferenceConstants.OUTLINE_LINK_WITH_EDITOR)) {
synchronizeOutlinePage(node, true);
}
setSelection(node, false);
}
}
static class TabConverter {
private int fTabRatio;
private ILineTracker fLineTracker;
public void setNumberOfSpacesPerTab(int ratio) {
fTabRatio= ratio;
}
public void setLineTracker(ILineTracker lineTracker) {
fLineTracker= lineTracker;
}
private int insertTabString(StringBuffer buffer, int offsetInLine) {
if (fTabRatio == 0) {
return 0;
}
int remainder= offsetInLine % fTabRatio;
remainder= fTabRatio - remainder;
for (int i= 0; i < remainder; i++) {
buffer.append(' ');
}
return remainder;
}
public void customizeDocumentCommand(IDocument document, DocumentCommand command) {
String text= command.text;
if (text == null) {
return;
}
int index= text.indexOf('\t');
if (index > -1) {
StringBuffer buffer= new StringBuffer();
fLineTracker.set(command.text);
int lines= fLineTracker.getNumberOfLines();
try {
for (int i= 0; i < lines; i++) {
int offset= fLineTracker.getLineOffset(i);
int endOffset= offset + fLineTracker.getLineLength(i);
String line= text.substring(offset, endOffset);
int position= 0;
if (i == 0) {
IRegion firstLine= document.getLineInformationOfOffset(command.offset);
position= command.offset - firstLine.getOffset();
}
int length= line.length();
for (int j= 0; j < length; j++) {
char c= line.charAt(j);
if (c == '\t') {
position += insertTabString(buffer, position);
} else {
buffer.append(c);
++ position;
}
}
}
command.text= buffer.toString();
} catch (BadLocationException x) {
}
}
}
}
class StatusLineSourceViewer extends SourceViewer{
private boolean fIgnoreTextConverters= false;
public StatusLineSourceViewer(Composite composite, IVerticalRuler verticalRuler, IOverviewRuler overviewRuler, int styles) {
super(composite, verticalRuler, overviewRuler, isOverviewRulerVisible(), styles);
}
/* (non-Javadoc)
* @see org.eclipse.jface.text.ITextOperationTarget#doOperation(int)
*/
public void doOperation(int operation) {
if (getTextWidget() == null || !redraws()) {
return;
}
switch (operation) {
case CONTENTASSIST_PROPOSALS:
String msg= fContentAssistant.showPossibleCompletions();
setStatusLineErrorMessage(msg);
return;
case UNDO:
fIgnoreTextConverters= true;
break;
case REDO:
fIgnoreTextConverters= true;
break;
}
super.doOperation(operation);
}
public void setTextConverter(TabConverter tabConverter) {
fTabConverter= tabConverter;
}
public void updateIndentationPrefixes() {
SourceViewerConfiguration configuration= getSourceViewerConfiguration();
String[] types= configuration.getConfiguredContentTypes(this);
for (int i= 0; i < types.length; i++) {
String[] prefixes= configuration.getIndentPrefixes(this, types[i]);
if (prefixes != null && prefixes.length > 0) {
setIndentPrefixes(prefixes, types[i]);
}
}
}
/* (non-Javadoc)
* @see org.eclipse.jface.text.TextViewer#customizeDocumentCommand(org.eclipse.jface.text.DocumentCommand)
*/
protected void customizeDocumentCommand(DocumentCommand command) {
super.customizeDocumentCommand(command);
if (!fIgnoreTextConverters && fTabConverter != null) {
fTabConverter.customizeDocumentCommand(getDocument(), command);
}
fIgnoreTextConverters= false;
}
}
/**
* Selection changed listener for the outline view.
*/
protected ISelectionChangedListener fSelectionChangedListener = new ISelectionChangedListener(){
public void selectionChanged(SelectionChangedEvent event) {
doSelectionChanged(event);
}
};
/**
* The page that shows the outline.
*/
protected AntEditorContentOutlinePage fOutlinePage;
/** The editor's tab to spaces converter */
private TabConverter fTabConverter;
private boolean fInitialReconcile= true;
/**
* The editor selection changed listener.
*
* @since 3.0
*/
private EditorSelectionChangedListener fEditorSelectionChangedListener;
public AntEditor() {
super();
setSourceViewerConfiguration(new AntEditorSourceViewerConfiguration(this));
setDocumentProvider(new AntEditorDocumentProvider(XMLCore.getDefault()));
JFaceResources.getColorRegistry().addListener(fColorChangeListener);
//TODO the framework does not currently support/listen to the color registry
//https://bugs.eclipse.org/bugs/show_bug.cgi?id=54554
// PreferenceConverter.setValue(getPreferenceStore(), ExtendedTextEditorPreferenceConstants.EDITOR_CURRENT_LINE_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.CURRENT_LINE_COLOR));
// PreferenceConverter.setValue(getPreferenceStore(), ExtendedTextEditorPreferenceConstants.EDITOR_LINE_NUMBER_RULER_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.LINE_NUMBER_RULER_COLOR));
// PreferenceConverter.setValue(getPreferenceStore(), ExtendedTextEditorPreferenceConstants.EDITOR_PRINT_MARGIN_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.PRINT_MARGIN_COLOR));
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#createActions()
*/
protected void createActions() {
super.createActions();
ResourceBundle bundle = ResourceBundle.getBundle("org.eclipse.ant.internal.ui.editor.AntEditorMessages"); //$NON-NLS-1$
IAction action = new ContentAssistAction(bundle, "ContentAssistProposal.", this); //$NON-NLS-1$
// This action definition is associated with the accelerator Ctrl+Space
action.setActionDefinitionId(ITextEditorActionDefinitionIds.CONTENT_ASSIST_PROPOSALS);
setAction("ContentAssistProposal", action); //$NON-NLS-1$
action = new TextOperationAction(bundle, "ContentFormat.", this, ISourceViewer.FORMAT); //$NON-NLS-1$
action.setActionDefinitionId(IJavaEditorActionDefinitionIds.FORMAT);
setAction("ContentFormat", action); //$NON-NLS-1$
//TODO set help
//WorkbenchHelp.setHelp(action, IJavaHelpContextIds.FORMAT_ACTION);
}
/*
* (non-Javadoc)
* @see org.eclipse.ui.editors.text.TextEditor#initializeEditor()
* Called from TextEditor.<init>
*/
protected void initializeEditor() {
super.initializeEditor();
setPreferenceStore(createCombinedPreferenceStore());
setCompatibilityMode(false);
setHelpContextId(IAntUIHelpContextIds.ANT_EDITOR);
}
/* (non-Javadoc)
* @see org.eclipse.core.runtime.IAdaptable#getAdapter(java.lang.Class)
*/
public Object getAdapter(Class key) {
if (key.equals(IContentOutlinePage.class)) {
return getOutlinePage();
}
return super.getAdapter(key);
}
private AntEditorContentOutlinePage getOutlinePage() {
if (fOutlinePage == null) {
fOutlinePage= new AntEditorContentOutlinePage(XMLCore.getDefault(), this);
fOutlinePage.addPostSelectionChangedListener(fSelectionChangedListener);
setOutlinePageInput(getEditorInput());
}
return fOutlinePage;
}
private void doSelectionChanged(SelectionChangedEvent selectionChangedEvent) {
IStructuredSelection selection= (IStructuredSelection)selectionChangedEvent.getSelection();
if (!isActivePart() && AntUIPlugin.getActivePage() != null) {
AntUIPlugin.getActivePage().bringToTop(this);
}
AntElementNode selectedXmlElement = (AntElementNode)selection.getFirstElement();
if(selectedXmlElement != null) {
setSelection(selectedXmlElement, !isActivePart());
}
}
private boolean isActivePart() {
IWorkbenchPart part= getActivePart();
return part != null && part.equals(this);
}
private void setSelection(AntElementNode reference, boolean moveCursor) {
if (reference != null) {
while (reference.getImportNode() != null) {
reference= reference.getImportNode();
}
if (reference.isExternal()) {
return;
}
StyledText textWidget= null;
ISourceViewer sourceViewer= getSourceViewer();
if (sourceViewer != null) {
textWidget= sourceViewer.getTextWidget();
}
if (textWidget == null) {
return;
}
try {
int offset= reference.getOffset();
int length= reference.getSelectionLength();
int highLightLength= reference.getLength();
if (offset < 0) {
return;
}
textWidget.setRedraw(false);
if (highLightLength > 0) {
setHighlightRange(offset, highLightLength, moveCursor);
}
if (!moveCursor) {
return;
}
if (offset > -1 && length > 0) {
sourceViewer.revealRange(offset, length);
// Selected region begins one index after offset
sourceViewer.setSelectedRange(offset, length);
}
} catch (IllegalArgumentException x) {
AntUIPlugin.log(x);
} finally {
if (textWidget != null) {
textWidget.setRedraw(true);
}
}
} else if (moveCursor) {
resetHighlightRange();
}
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#affectsTextPresentation(org.eclipse.jface.util.PropertyChangeEvent)
*/
protected boolean affectsTextPresentation(PropertyChangeEvent event) {
String property= event.getProperty();
return property.equals(IAntEditorColorConstants.TEXT_COLOR) ||
property.equals(IAntEditorColorConstants.PROCESSING_INSTRUCTIONS_COLOR) ||
property.equals(IAntEditorColorConstants.STRING_COLOR) ||
property.equals(IAntEditorColorConstants.TAG_COLOR) ||
property.equals(IAntEditorColorConstants.XML_COMMENT_COLOR);
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#handlePreferenceStoreChanged(org.eclipse.jface.util.PropertyChangeEvent)
*/
protected void handlePreferenceStoreChanged(PropertyChangeEvent event) {
String property= event.getProperty();
if (AbstractDecoratedTextEditorPreferenceConstants.EDITOR_TAB_WIDTH.equals(property)) {
Object value= event.getNewValue();
StatusLineSourceViewer viewer= (StatusLineSourceViewer) getSourceViewer();
if (value instanceof Integer) {
viewer.getTextWidget().setTabs(((Integer) value).intValue());
} else if (value instanceof String) {
viewer.getTextWidget().setTabs(Integer.parseInt((String) value));
}
return;
}
if (AntEditorPreferenceConstants.EDITOR_SPACES_FOR_TABS.equals(property)) {
if (isTabConversionEnabled()) {
startTabConversion();
} else {
stopTabConversion();
}
return;
}
AntEditorSourceViewerConfiguration sourceViewerConfiguration= (AntEditorSourceViewerConfiguration)getSourceViewerConfiguration();
if (affectsTextPresentation(event)) {
sourceViewerConfiguration.updateScanners();
}
sourceViewerConfiguration.changeConfiguration(event);
//TODO the framework does not currently support/listen to the color registry
//https://bugs.eclipse.org/bugs/show_bug.cgi?id=54554
if (property.equals(AntEditorPreferenceConstants.CURRENT_LINE_COLOR)) {
PreferenceConverter.setValue(getPreferenceStore(), AbstractDecoratedTextEditorPreferenceConstants.EDITOR_CURRENT_LINE_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.CURRENT_LINE_COLOR));
return;
} else if (property.equals(AntEditorPreferenceConstants.LINE_NUMBER_RULER_COLOR)) {
PreferenceConverter.setValue(getPreferenceStore(), AbstractDecoratedTextEditorPreferenceConstants.EDITOR_LINE_NUMBER_RULER_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.LINE_NUMBER_RULER_COLOR));
return;
} else if (property.equals(AntEditorPreferenceConstants.PRINT_MARGIN_COLOR)) {
PreferenceConverter.setValue(getPreferenceStore(), AbstractDecoratedTextEditorPreferenceConstants.EDITOR_PRINT_MARGIN_COLOR, JFaceResources.getColorRegistry().getRGB(AntEditorPreferenceConstants.PRINT_MARGIN_COLOR));
return;
}
super.handlePreferenceStoreChanged(event);
}
/*
* @see org.eclipse.ui.editors.text.TextEditor#doSetInput(org.eclipse.ui.IEditorInput)
*/
protected void doSetInput(IEditorInput input) throws CoreException {
super.doSetInput(input);
setOutlinePageInput(input);
}
private void setOutlinePageInput(IEditorInput input) {
if (fOutlinePage != null) {
IDocumentProvider provider= getDocumentProvider();
if (provider instanceof AntEditorDocumentProvider) {
AntEditorDocumentProvider documentProvider= (AntEditorDocumentProvider) provider;
AntModel model= documentProvider.getAntModel(input);
fOutlinePage.setPageInput(model);
}
}
}
/**
* Returns the Ant model for the current editor input of this editor.
* @return the Ant model for this editor or <code>null</code>
*/
public AntModel getAntModel() {
IDocumentProvider provider= getDocumentProvider();
if (provider instanceof AntEditorDocumentProvider) {
AntEditorDocumentProvider documentProvider= (AntEditorDocumentProvider) provider;
return documentProvider.getAntModel(getEditorInput());
}
return null;
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.ExtendedTextEditor#createAnnotationAccess()
*/
protected IAnnotationAccess createAnnotationAccess() {
return new AnnotationAccess();
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#createSourceViewer(org.eclipse.swt.widgets.Composite, org.eclipse.jface.text.source.IVerticalRuler, int)
*/
protected ISourceViewer createSourceViewer(Composite parent, IVerticalRuler ruler, int styles) {
fOverviewRuler= createOverviewRuler(getSharedColors());
ISourceViewer viewer= new StatusLineSourceViewer(parent, ruler, getOverviewRuler(), styles);
//ensure decoration support has been created and configured.
getSourceViewerDecorationSupport(viewer);
return viewer;
}
/**
* Ses the given message as error message to this editor's status line.
* @param msg message to be set
*/
protected void setStatusLineErrorMessage(String msg) {
IEditorStatusLine statusLine= (IEditorStatusLine) getAdapter(IEditorStatusLine.class);
if (statusLine != null)
statusLine.setMessage(true, msg, null);
}
public void openReferenceElement() {
ISelection selection= getSelectionProvider().getSelection();
String errorMessage= null;
if (selection instanceof ITextSelection) {
ITextSelection textSelection= (ITextSelection)selection;
String text= textSelection.getText();
AntElementNode node= getAntModel().getReferenceNode(text);
if (node != null) {
setSelection(node, true);
return;
}
String path= getAntModel().getEntityPath(text);
if (path != null) {
IFile file= AntUtil.getFileForLocation(path, null);
if (file.exists()) {
try {
openInEditor(file, isActivePart());
return;
} catch (PartInitException e) {
errorMessage= e.getLocalizedMessage();
}
}
}
}
if (errorMessage == null) {
errorMessage= AntEditorMessages.getString("AntEditor.3"); //$NON-NLS-1$
}
setStatusLineErrorMessage(errorMessage);
getSite().getShell().getDisplay().beep();
}
private void openInEditor(IFile file, boolean activate) throws PartInitException {
if (file != null) {
IWorkbenchPage p= getEditorSite().getPage();
if (p != null) {
IDE.openEditor(p, file, activate);
}
}
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#editorContextMenuAboutToShow(org.eclipse.jface.action.IMenuManager)
*/
public void editorContextMenuAboutToShow(IMenuManager menu) {
super.editorContextMenuAboutToShow(menu);
IAction formatAction= getAction("ContentFormat"); //$NON-NLS-1$
if (formatAction == null) {
return;
}
if (formatAction.isEnabled()) {
menu.add(formatAction);
}
}
private void startTabConversion() {
if (fTabConverter == null) {
fTabConverter= new TabConverter();
fTabConverter.setLineTracker(new DefaultLineTracker());
fTabConverter.setNumberOfSpacesPerTab(getTabSize());
StatusLineSourceViewer viewer= (StatusLineSourceViewer) getSourceViewer();
viewer.setTextConverter(fTabConverter);
// http://dev.eclipse.org/bugs/show_bug.cgi?id=19270
viewer.updateIndentationPrefixes();
}
}
private void stopTabConversion() {
if (fTabConverter != null) {
StatusLineSourceViewer viewer= (StatusLineSourceViewer) getSourceViewer();
viewer.setTextConverter(null);
// http://dev.eclipse.org/bugs/show_bug.cgi?id=19270
viewer.updateIndentationPrefixes();
fTabConverter= null;
}
}
private int getTabSize() {
IPreferenceStore preferences= getPreferenceStore();
return preferences.getInt(AbstractDecoratedTextEditorPreferenceConstants.EDITOR_TAB_WIDTH);
}
/* (non-Javadoc)
* @see org.eclipse.ui.IWorkbenchPart#createPartControl(org.eclipse.swt.widgets.Composite)
*/
public void createPartControl(Composite parent) {
super.createPartControl(parent);
if (isTabConversionEnabled()) {
startTabConversion();
}
fEditorSelectionChangedListener= new EditorSelectionChangedListener();
fEditorSelectionChangedListener.install(getSelectionProvider());
}
private boolean isTabConversionEnabled() {
IPreferenceStore store= getPreferenceStore();
return store.getBoolean(AntEditorPreferenceConstants.EDITOR_SPACES_FOR_TABS);
}
/* (non-Javadoc)
* @see org.eclipse.ui.IWorkbenchPart#dispose()
*/
public void dispose() {
super.dispose();
JFaceResources.getColorRegistry().removeListener(fColorChangeListener);
if (fEditorSelectionChangedListener != null) {
fEditorSelectionChangedListener.uninstall(getSelectionProvider());
fEditorSelectionChangedListener= null;
}
}
/* (non-Javadoc)
* @see org.eclipse.ui.ISaveablePart#doSave(org.eclipse.core.runtime.IProgressMonitor)
*/
public void doSave(IProgressMonitor monitor) {
super.doSave(monitor);
getAntModel().updateMarkers();
updateEditorImage();
}
private void updateEditorImage() {
Image titleImage= getTitleImage();
if (titleImage == null) {
return;
}
AntProjectNode node= getAntModel().getProjectNode();
if (node != null) {
postImageChange(node);
}
}
private void updateForInitialReconcile() {
if (getAntModel() == null) {
return;
}
fInitialReconcile= false;
updateEditorImage();
getAntModel().updateForInitialReconcile();
}
private void postImageChange(final AntElementNode node) {
Shell shell= getSite().getShell();
if (shell != null && !shell.isDisposed()) {
shell.getDisplay().asyncExec(new Runnable() {
public void run() {
if (getSite().getShell() == null || getSite().getShell().isDisposed()) {
return;
}
Image titleImage= getTitleImage();
Image newImage= node.getImage();
if (titleImage != newImage) {
setTitleImage(newImage);
}
}
});
}
}
public void synchronizeOutlinePage(boolean checkIfOutlinePageActive) {
if (getSelectionProvider() == null) {
return;
}
AntElementNode node= getNode();
synchronizeOutlinePage(node, checkIfOutlinePageActive);
}
protected void synchronize(boolean checkIfOutlinePageActive) {
if (getSelectionProvider() == null) {
return;
}
AntElementNode node= getNode();
if (AntUIPlugin.getDefault().getPreferenceStore().getBoolean(IAntUIPreferenceConstants.OUTLINE_LINK_WITH_EDITOR)) {
synchronizeOutlinePage(node, checkIfOutlinePageActive);
}
setSelection(node, false);
}
private AntElementNode getNode() {
AntModel model= getAntModel();
if (model == null) {
return null;
}
AntElementNode node= null;
ISelection selection= getSelectionProvider().getSelection();
if (selection instanceof ITextSelection) {
ITextSelection textSelection= (ITextSelection)selection;
int offset= textSelection.getOffset();
node= model.getNode(offset, false);
}
return node;
}
protected void synchronizeOutlinePage(AntElementNode node, boolean checkIfOutlinePageActive) {
if (fOutlinePage != null && !(checkIfOutlinePageActive && isAntOutlinePageAction())) {
fOutlinePage.removePostSelectionChangedListener(fSelectionChangedListener);
fOutlinePage.select(node);
fOutlinePage.addPostSelectionChangedListener(fSelectionChangedListener);
}
}
/* (non-Javadoc)
* @see org.eclipse.ant.internal.ui.editor.text.IReconcilingParticipant#reconciled()
*/
public void reconciled() {
if (getSourceViewerConfiguration() == null) {
return; //editor has been disposed.
}
if (fInitialReconcile) {
updateForInitialReconcile();
}
IAutoIndentStrategy strategy= getSourceViewerConfiguration().getAutoIndentStrategy(null, null);
if (strategy instanceof AntAutoIndentStrategy) {
((AntAutoIndentStrategy)strategy).reconciled();
}
Shell shell= getSite().getShell();
if (shell != null && !shell.isDisposed()) {
shell.getDisplay().asyncExec(new Runnable() {
public void run() {
if (getSite().getShell() == null || getSite().getShell().isDisposed()) {
return;
}
synchronize(true);
}
});
}
}
private boolean isAntOutlinePageAction() {
IWorkbenchPart part= getActivePart();
return part instanceof ContentOutline && ((ContentOutline)part).getCurrentPage() == fOutlinePage;
}
private IWorkbenchPart getActivePart() {
IWorkbenchWindow window= getSite().getWorkbenchWindow();
IPartService service= window.getPartService();
return service.getActivePart();
}
/* (non-Javadoc)
* @see org.eclipse.ui.texteditor.AbstractTextEditor#doSetSelection(org.eclipse.jface.viewers.ISelection)
*/
protected void doSetSelection(ISelection selection) {
super.doSetSelection(selection);
synchronizeOutlinePage(true);
}
/**
* Creates a combined preference store, this store is read-only.
*
* @return the combined preference store
* @since 3.0
*/
private IPreferenceStore createCombinedPreferenceStore() {
IPreferenceStore antStore= AntUIPlugin.getDefault().getPreferenceStore();
IPreferenceStore generalTextStore= EditorsUI.getPreferenceStore();
return new ChainedPreferenceStore(new IPreferenceStore[] { antStore, generalTextStore });
}
}