/******************************************************************************* * Copyright (c) 2000, 2005 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.rubypeople.rdt.internal.ui.text.ruby; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.ISafeRunnable; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.NullProgressMonitor; import org.eclipse.core.runtime.OperationCanceledException; import org.eclipse.core.runtime.Platform; import org.eclipse.core.runtime.Status; import org.eclipse.jface.text.Assert; import org.eclipse.jface.text.IDocument; import org.eclipse.jface.text.IRegion; import org.eclipse.jface.text.reconciler.DirtyRegion; import org.eclipse.jface.text.reconciler.IReconcilingStrategy; import org.eclipse.jface.text.reconciler.IReconcilingStrategyExtension; import org.eclipse.jface.text.source.IAnnotationModel; import org.eclipse.ui.texteditor.IDocumentProvider; import org.eclipse.ui.texteditor.ITextEditor; import org.jruby.ast.RootNode; import org.rubypeople.rdt.core.IRubyScript; import org.rubypeople.rdt.core.RubyModelException; import org.rubypeople.rdt.internal.ui.RubyPlugin; import org.rubypeople.rdt.ui.IWorkingCopyManager; import org.rubypeople.rdt.ui.RubyUI; public class RubyReconcilingStrategy implements IReconcilingStrategy, IReconcilingStrategyExtension { private ITextEditor fEditor; private IWorkingCopyManager fManager; private IDocumentProvider fDocumentProvider; private IProgressMonitor fProgressMonitor; private boolean fNotify= true; private IRubyReconcilingListener fRubyReconcilingListener; private boolean fIsRubyReconcilingListener; public RubyReconcilingStrategy(ITextEditor editor) { fEditor= editor; fManager= RubyPlugin.getDefault().getWorkingCopyManager(); fDocumentProvider= RubyPlugin.getDefault().getRubyDocumentProvider(); fIsRubyReconcilingListener= fEditor instanceof IRubyReconcilingListener; if (fIsRubyReconcilingListener) fRubyReconcilingListener= (IRubyReconcilingListener)fEditor; } private IProblemRequestorExtension getProblemRequestorExtension() { IAnnotationModel model= fDocumentProvider.getAnnotationModel(fEditor.getEditorInput()); if (model instanceof IProblemRequestorExtension) return (IProblemRequestorExtension) model; return null; } private void reconcile(final boolean initialReconcile) { final RootNode[] ast= new RootNode[1]; final IRubyScript unit= fManager.getWorkingCopy(fEditor.getEditorInput()); try { if (unit != null) { Platform.run(new ISafeRunnable() { public void run() { try { /* fix for missing cancel flag communication */ IProblemRequestorExtension extension= getProblemRequestorExtension(); if (extension != null) { extension.setProgressMonitor(fProgressMonitor); extension.setIsActive(true); } try { ast[0] = unit.reconcile(true, null, fProgressMonitor); } catch (OperationCanceledException ex) { Assert.isTrue(fProgressMonitor == null || fProgressMonitor.isCanceled()); ast[0]= null; } finally { /* fix for missing cancel flag communication */ if (extension != null) { extension.setProgressMonitor(null); extension.setIsActive(false); } } } catch (RubyModelException ex) { handleException(ex); } } public void handleException(Throwable ex) { IStatus status= new Status(IStatus.ERROR, RubyUI.ID_PLUGIN, IStatus.OK, "Error in RDT Core during reconcile", ex); //$NON-NLS-1$ RubyPlugin.getDefault().getLog().log(status); } }); } } finally { // Always notify listeners, see https://bugs.eclipse.org/bugs/show_bug.cgi?id=55969 for the final solution try { if (fIsRubyReconcilingListener) { IProgressMonitor pm= fProgressMonitor; if (pm == null) pm= new NullProgressMonitor(); fRubyReconcilingListener.reconciled(unit, ast[0], !fNotify, pm); } } finally { fNotify= true; } } } /* * @see IReconcilingStrategy#reconcile(IRegion) */ public void reconcile(IRegion partition) { reconcile(false); } /* * @see IReconcilingStrategy#reconcile(DirtyRegion, IRegion) */ public void reconcile(DirtyRegion dirtyRegion, IRegion subRegion) { reconcile(false); } /* * @see IReconcilingStrategy#setDocument(IDocument) */ public void setDocument(IDocument document) { } /* * @see IReconcilingStrategyExtension#setProgressMonitor(IProgressMonitor) */ public void setProgressMonitor(IProgressMonitor monitor) { fProgressMonitor= monitor; } /* * @see IReconcilingStrategyExtension#initialReconcile() */ public void initialReconcile() { reconcile(true); } /** * Tells this strategy whether to inform its listeners. * * @param notify <code>true</code> if listeners should be notified */ public void notifyListeners(boolean notify) { fNotify= notify; } /** * Called before reconciling is started. * * @since 3.0 */ public void aboutToBeReconciled() { if (fIsRubyReconcilingListener) fRubyReconcilingListener.aboutToBeReconciled(); } }