| /******************************************************************************* |
| * Copyright (c) 2000, 2018 IBM Corporation and others. |
| * |
| * This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License 2.0 |
| * which accompanies this distribution, and is available at |
| * https://www.eclipse.org/legal/epl-2.0/ |
| * |
| * SPDX-License-Identifier: EPL-2.0 |
| * |
| * Contributors: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.jface.text; |
| |
| |
| import org.eclipse.swt.custom.StyledText; |
| import org.eclipse.swt.events.MouseMoveListener; |
| import org.eclipse.swt.graphics.Point; |
| import org.eclipse.swt.graphics.Rectangle; |
| import org.eclipse.swt.widgets.Display; |
| |
| import org.eclipse.core.runtime.ILog; |
| import org.eclipse.core.runtime.IStatus; |
| import org.eclipse.core.runtime.OperationCanceledException; |
| import org.eclipse.core.runtime.Platform; |
| import org.eclipse.core.runtime.Status; |
| |
| |
| /** |
| * This manager controls the layout, content, and visibility of an information |
| * control in reaction to mouse hover events issued by the text widget of a |
| * text viewer. It overrides <code>computeInformation</code>, so that the |
| * computation is performed in a dedicated background thread. This implies |
| * that the used <code>ITextHover</code> objects must be capable of |
| * operating in a non-UI thread. |
| * |
| * @since 2.0 |
| */ |
| class TextViewerHoverManager extends AbstractHoverInformationControlManager implements IWidgetTokenKeeper, IWidgetTokenKeeperExtension { |
| |
| |
| /** |
| * Priority of the hovers managed by this manager. |
| * Default value: <code>0</code>; |
| * @since 3.0 |
| */ |
| public final static int WIDGET_PRIORITY= 0; |
| |
| |
| /** The text viewer */ |
| private TextViewer fTextViewer; |
| /** The hover information computation thread */ |
| private Thread fThread; |
| /** The stopper of the computation thread */ |
| private ITextListener fStopper; |
| /** Internal monitor */ |
| private Object fMutex= new Object(); |
| /** The currently shown text hover. */ |
| private volatile ITextHover fTextHover; |
| /** |
| * Tells whether the next mouse hover event |
| * should be processed. |
| * @since 3.0 |
| */ |
| private boolean fProcessMouseHoverEvent= true; |
| /** |
| * Internal mouse move listener. |
| * @since 3.0 |
| */ |
| private MouseMoveListener fMouseMoveListener; |
| /** |
| * Internal view port listener. |
| * @since 3.0 |
| */ |
| private IViewportListener fViewportListener; |
| |
| |
| /** |
| * Creates a new text viewer hover manager specific for the given text viewer. |
| * The manager uses the given information control creator. |
| * |
| * @param textViewer the viewer for which the controller is created |
| * @param creator the information control creator |
| */ |
| public TextViewerHoverManager(TextViewer textViewer, IInformationControlCreator creator) { |
| super(creator); |
| fTextViewer= textViewer; |
| fStopper= event -> { |
| synchronized (fMutex) { |
| if (fThread != null) { |
| fThread.interrupt(); |
| fThread= null; |
| } |
| } |
| }; |
| fViewportListener= verticalOffset -> fProcessMouseHoverEvent= false; |
| fTextViewer.addViewportListener(fViewportListener); |
| fMouseMoveListener= event -> fProcessMouseHoverEvent= true; |
| fTextViewer.getTextWidget().addMouseMoveListener(fMouseMoveListener); |
| } |
| |
| /** |
| * Determines all necessary details and delegates the computation into |
| * a background thread. |
| */ |
| @Override |
| protected void computeInformation() { |
| |
| if (!fProcessMouseHoverEvent) { |
| setInformation(null, null); |
| return; |
| } |
| |
| Point location= getHoverEventLocation(); |
| int offset= computeOffsetAtLocation(location.x, location.y); |
| if (offset == -1) { |
| setInformation(null, null); |
| return; |
| } |
| |
| final ITextHover hover= fTextViewer.getTextHover(offset, getHoverEventStateMask()); |
| if (hover == null) { |
| setInformation(null, null); |
| return; |
| } |
| |
| final IRegion region= hover.getHoverRegion(fTextViewer, offset); |
| if (region == null) { |
| setInformation(null, null); |
| return; |
| } |
| |
| final Rectangle area= JFaceTextUtil.computeArea(region, fTextViewer); |
| if (area == null || area.isEmpty()) { |
| setInformation(null, null); |
| return; |
| } |
| |
| if (fThread != null) { |
| setInformation(null, null); |
| return; |
| } |
| |
| fThread= new Thread("Text Viewer Hover Presenter") { //$NON-NLS-1$ |
| @Override |
| public void run() { |
| // http://bugs.eclipse.org/bugs/show_bug.cgi?id=17693 |
| boolean hasFinished= false; |
| try { |
| if (fThread != null) { |
| Object information; |
| try { |
| if (hover instanceof ITextHoverExtension2) |
| information= ((ITextHoverExtension2)hover).getHoverInfo2(fTextViewer, region); |
| else |
| information= hover.getHoverInfo(fTextViewer, region); |
| } catch (ArrayIndexOutOfBoundsException x) { |
| /* |
| * This code runs in a separate thread which can |
| * lead to text offsets being out of bounds when |
| * computing the hover info (see bug 32848). |
| */ |
| information= null; |
| } |
| |
| if (hover instanceof ITextHoverExtension) |
| setCustomInformationControlCreator(((ITextHoverExtension) hover).getHoverControlCreator()); |
| else |
| setCustomInformationControlCreator(null); |
| |
| setInformation(information, area); |
| if (information != null) |
| fTextHover= hover; |
| } else { |
| setInformation(null, null); |
| } |
| hasFinished= true; |
| } catch (OperationCanceledException e) { |
| // Just swallow the exception if the operation was canceled |
| } catch (RuntimeException ex) { |
| String PLUGIN_ID= "org.eclipse.jface.text"; //$NON-NLS-1$ |
| ILog log= Platform.getLog(Platform.getBundle(PLUGIN_ID)); |
| log.log(new Status(IStatus.ERROR, PLUGIN_ID, IStatus.OK, "Unexpected runtime error while computing a text hover", ex)); //$NON-NLS-1$ |
| } finally { |
| synchronized (fMutex) { |
| if (fTextViewer != null) |
| fTextViewer.removeTextListener(fStopper); |
| fThread= null; |
| // https://bugs.eclipse.org/bugs/show_bug.cgi?id=44756 |
| if (!hasFinished) |
| setInformation(null, null); |
| } |
| } |
| } |
| }; |
| |
| fThread.setDaemon(true); |
| fThread.setPriority(Thread.MIN_PRIORITY); |
| synchronized (fMutex) { |
| fTextViewer.addTextListener(fStopper); |
| fThread.start(); |
| } |
| } |
| |
| /** |
| * As computation is done in the background, this method is |
| * also called in the background thread. Delegates the control |
| * flow back into the UI thread, in order to allow displaying the |
| * information in the information control. |
| */ |
| @Override |
| protected void presentInformation() { |
| if (fTextViewer == null) { |
| return; |
| } |
| |
| StyledText textWidget= fTextViewer.getTextWidget(); |
| if (textWidget != null && !textWidget.isDisposed()) { |
| Display display= textWidget.getDisplay(); |
| if (display == null) { |
| return; |
| } |
| |
| display.asyncExec(() -> { |
| if (!textWidget.isDisposed()) { |
| doPresentInformation(); |
| } |
| }); |
| } |
| } |
| |
| /* |
| * @see AbstractInformationControlManager#presentInformation() |
| */ |
| protected void doPresentInformation() { |
| super.presentInformation(); |
| } |
| |
| /** |
| * Computes the document offset underlying the given text widget coordinates. |
| * This method uses a linear search as it cannot make any assumption about |
| * how the document is actually presented in the widget. (Covers cases such |
| * as bidirectional text.) |
| * |
| * @param x the horizontal coordinate inside the text widget |
| * @param y the vertical coordinate inside the text widget |
| * @return the document offset corresponding to the given point |
| */ |
| private int computeOffsetAtLocation(int x, int y) { |
| try { |
| StyledText styledText= fTextViewer.getTextWidget(); |
| int widgetOffset= styledText.getOffsetAtPoint(new Point(x, y)); |
| if (widgetOffset == -1) { |
| return -1; |
| } |
| Point p= styledText.getLocationAtOffset(widgetOffset); |
| if (p.x > x) { |
| widgetOffset--; |
| } |
| |
| if (fTextViewer instanceof ITextViewerExtension5) { |
| ITextViewerExtension5 extension= (ITextViewerExtension5) fTextViewer; |
| return extension.widgetOffset2ModelOffset(widgetOffset); |
| } |
| |
| return widgetOffset + fTextViewer._getVisibleRegionOffset(); |
| } catch (IllegalArgumentException e) { |
| return -1; |
| } |
| } |
| |
| @Override |
| protected void showInformationControl(Rectangle subjectArea) { |
| if (fTextViewer != null && fTextViewer.requestWidgetToken(this, WIDGET_PRIORITY)) |
| super.showInformationControl(subjectArea); |
| else |
| if (DEBUG) |
| System.out.println("TextViewerHoverManager#showInformationControl(..) did not get widget token"); //$NON-NLS-1$ |
| } |
| |
| @Override |
| protected void hideInformationControl() { |
| try { |
| fTextHover= null; |
| super.hideInformationControl(); |
| } finally { |
| if (fTextViewer != null) |
| fTextViewer.releaseWidgetToken(this); |
| } |
| } |
| |
| @Override |
| void replaceInformationControl(boolean takeFocus) { |
| if (fTextViewer != null) |
| fTextViewer.releaseWidgetToken(this); |
| super.replaceInformationControl(takeFocus); |
| } |
| |
| @Override |
| protected void handleInformationControlDisposed() { |
| try { |
| super.handleInformationControlDisposed(); |
| } finally { |
| if (fTextViewer != null) |
| fTextViewer.releaseWidgetToken(this); |
| } |
| } |
| |
| @Override |
| public boolean requestWidgetToken(IWidgetTokenOwner owner) { |
| fTextHover= null; |
| super.hideInformationControl(); |
| return true; |
| } |
| |
| @Override |
| public boolean requestWidgetToken(IWidgetTokenOwner owner, int priority) { |
| if (priority > WIDGET_PRIORITY) { |
| fTextHover= null; |
| super.hideInformationControl(); |
| return true; |
| } |
| return false; |
| } |
| |
| @Override |
| public boolean setFocus(IWidgetTokenOwner owner) { |
| if (! hasInformationControlReplacer()) |
| return false; |
| |
| IInformationControl iControl= getCurrentInformationControl(); |
| if (canReplace(iControl)) { |
| if (cancelReplacingDelay()) |
| replaceInformationControl(true); |
| |
| return true; |
| } |
| if (iControl instanceof IInformationControlExtension5) { |
| return true; // The iControl didn't return an information presenter control creator, so let's stop here. |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Returns the currently shown text hover or <code>null</code> if no text |
| * hover is shown. |
| * |
| * @return the currently shown text hover or <code>null</code> |
| */ |
| protected ITextHover getCurrentTextHover() { |
| return fTextHover; |
| } |
| |
| @Override |
| public void dispose() { |
| if (fTextViewer != null) { |
| fTextViewer.removeViewportListener(fViewportListener); |
| fViewportListener= null; |
| |
| StyledText st= fTextViewer.getTextWidget(); |
| if (st != null && !st.isDisposed()) |
| st.removeMouseMoveListener(fMouseMoveListener); |
| fMouseMoveListener= null; |
| } |
| super.dispose(); |
| } |
| } |