diff dwtx/jface/internal/text/html/BrowserInformationControl.d @ 129:eb30df5ca28b

Added JFace Text sources
author Frank Benoit <benoit@tionex.de>
date Sat, 23 Aug 2008 19:10:48 +0200
parents
children c4fb132a086c
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dwtx/jface/internal/text/html/BrowserInformationControl.d	Sat Aug 23 19:10:48 2008 +0200
@@ -0,0 +1,624 @@
+/*******************************************************************************
+ * Copyright (c) 2000, 2008 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
+ * Port to the D programming language:
+ *     Frank Benoit <benoit@tionex.de>
+ *******************************************************************************/
+module dwtx.jface.internal.text.html.BrowserInformationControl;
+
+import dwt.dwthelper.utils;
+
+import java.io.IOException;
+import java.io.StringReader;
+import java.util.Iterator;
+
+import dwt.DWT;
+import dwt.DWTError;
+import dwt.browser.Browser;
+import dwt.browser.LocationListener;
+import dwt.browser.ProgressAdapter;
+import dwt.browser.ProgressEvent;
+import dwt.custom.StyleRange;
+import dwt.events.KeyEvent;
+import dwt.events.KeyListener;
+import dwt.graphics.Color;
+import dwt.graphics.Font;
+import dwt.graphics.FontData;
+import dwt.graphics.GC;
+import dwt.graphics.Point;
+import dwt.graphics.Rectangle;
+import dwt.graphics.TextLayout;
+import dwt.graphics.TextStyle;
+import dwt.widgets.Composite;
+import dwt.widgets.Display;
+import dwt.widgets.Menu;
+import dwt.widgets.Shell;
+import dwt.widgets.Slider;
+import dwtx.core.runtime.Assert;
+import dwtx.core.runtime.ListenerList;
+import dwtx.jface.action.ToolBarManager;
+import dwtx.jface.resource.JFaceResources;
+import dwtx.jface.text.AbstractInformationControl;
+import dwtx.jface.text.IDelayedInputChangeProvider;
+import dwtx.jface.text.IInformationControlExtension2;
+import dwtx.jface.text.IInputChangedListener;
+import dwtx.jface.text.TextPresentation;
+
+
+/**
+ * Displays HTML information in a {@link dwt.browser.Browser} widget.
+ * <p>
+ * This {@link IInformationControlExtension2} expects {@link #setInput(Object)} to be
+ * called with an argument of type {@link BrowserInformationControlInput}.
+ * </p>
+ * <p>
+ * Moved into this package from <code>dwtx.jface.internal.text.revisions</code>.</p>
+ * <p>
+ * This class may be instantiated; it is not intended to be subclassed.</p>
+ * <p>
+ * Current problems:
+ * <ul>
+ *  <li>the size computation is too small</li>
+ *  <li>focusLost event is not sent - see https://bugs.eclipse.org/bugs/show_bug.cgi?id=84532</li>
+ * </ul>
+ * </p>
+ * 
+ * @since 3.2
+ */
+public class BrowserInformationControl : AbstractInformationControl , IInformationControlExtension2, IDelayedInputChangeProvider {
+
+
+    /**
+     * Tells whether the DWT Browser widget and hence this information
+     * control is available.
+     *
+     * @param parent the parent component used for checking or <code>null</code> if none
+     * @return <code>true</code> if this control is available
+     */
+    public static bool isAvailable(Composite parent) {
+        if (!fgAvailabilityChecked) {
+            try {
+                Browser browser= new Browser(parent, DWT.NONE);
+                browser.dispose();
+                fgIsAvailable= true;
+                
+                Slider sliderV= new Slider(parent, DWT.VERTICAL);
+                Slider sliderH= new Slider(parent, DWT.HORIZONTAL);
+                int width= sliderV.computeSize(DWT.DEFAULT, DWT.DEFAULT).x;
+                int height= sliderH.computeSize(DWT.DEFAULT, DWT.DEFAULT).y;
+                fgScrollBarSize= new Point(width, height);
+                sliderV.dispose();
+                sliderH.dispose();
+            } catch (DWTError er) {
+                fgIsAvailable= false;
+            } finally {
+                fgAvailabilityChecked= true;
+            }
+        }
+
+        return fgIsAvailable;
+    }
+
+
+    /**
+     * Minimal size constraints.
+     * @since 3.2
+     */
+    private static final int MIN_WIDTH= 80;
+    private static final int MIN_HEIGHT= 50;
+
+    
+    /**
+     * Availability checking cache.
+     */
+    private static bool fgIsAvailable= false;
+    private static bool fgAvailabilityChecked= false;
+
+    /**
+     * Cached scroll bar width and height
+     * @since 3.4
+     */
+    private static Point fgScrollBarSize;
+    
+    /** The control's browser widget */
+    private Browser fBrowser;
+    /** Tells whether the browser has content */
+    private bool fBrowserHasContent;
+    /** Text layout used to approximate size of content when rendered in browser */
+    private TextLayout fTextLayout;
+    /** Bold text style */
+    private TextStyle fBoldStyle;
+
+    private BrowserInformationControlInput fInput;
+
+    /**
+     * <code>true</code> iff the browser has completed loading of the last
+     * input set via {@link #setInformation(String)}.
+     * @since 3.4
+     */
+    private bool fCompleted= false;
+
+    /**
+     * The listener to be notified when a delayed location changing event happened.
+     * @since 3.4
+     */
+    private IInputChangedListener fDelayedInputChangeListener;
+
+    /**
+     * The listeners to be notified when the input changed.
+     * @since 3.4
+     */
+    private ListenerList/*<IInputChangedListener>*/ fInputChangeListeners= new ListenerList(ListenerList.IDENTITY);
+
+    /**
+     * The symbolic name of the font used for size computations, or <code>null</code> to use dialog font.
+     * @since 3.4
+     */
+    private final String fSymbolicFontName;
+
+
+    /**
+     * Creates a browser information control with the given shell as parent.
+     * 
+     * @param parent the parent shell
+     * @param symbolicFontName the symbolic name of the font used for size computations
+     * @param resizable <code>true</code> if the control should be resizable
+     * @since 3.4
+     */
+    public BrowserInformationControl(Shell parent, String symbolicFontName, bool resizable) {
+        super(parent, resizable);
+        fSymbolicFontName= symbolicFontName;
+        create();
+    }
+    
+    /**
+     * Creates a browser information control with the given shell as parent.
+     * 
+     * @param parent the parent shell
+     * @param symbolicFontName the symbolic name of the font used for size computations
+     * @param statusFieldText the text to be used in the optional status field
+     *            or <code>null</code> if the status field should be hidden
+     * @since 3.4
+     */
+    public BrowserInformationControl(Shell parent, String symbolicFontName, String statusFieldText) {
+        super(parent, statusFieldText);
+        fSymbolicFontName= symbolicFontName;
+        create();
+    }
+    
+    /**
+     * Creates a browser information control with the given shell as parent.
+     * 
+     * @param parent the parent shell
+     * @param symbolicFontName the symbolic name of the font used for size computations
+     * @param toolBarManager the manager or <code>null</code> if toolbar is not desired
+     * @since 3.4
+     */
+    public BrowserInformationControl(Shell parent, String symbolicFontName, ToolBarManager toolBarManager) {
+        super(parent, toolBarManager);
+        fSymbolicFontName= symbolicFontName;
+        create();
+    }
+    
+    /*
+     * @see dwtx.jface.text.AbstractInformationControl#createContent(dwt.widgets.Composite)
+     */
+    protected void createContent(Composite parent) {
+        fBrowser= new Browser(parent, DWT.NONE);
+        
+        Display display= getShell().getDisplay();
+        fBrowser.setForeground(display.getSystemColor(DWT.COLOR_INFO_FOREGROUND));
+        fBrowser.setBackground(display.getSystemColor(DWT.COLOR_INFO_BACKGROUND));
+        fBrowser.addKeyListener(new KeyListener() {
+
+            public void keyPressed(KeyEvent e)  {
+                if (e.character is 0x1B) // ESC
+                    getShell().dispose(); // XXX: Just hide? Would avoid constant recreations.
+            }
+
+            public void keyReleased(KeyEvent e) {}
+        });
+
+        fBrowser.addProgressListener(new ProgressAdapter() {
+            public void completed(ProgressEvent event) {
+                fCompleted= true;
+            }
+        });
+        
+        // Replace browser's built-in context menu with none
+        fBrowser.setMenu(new Menu(getShell(), DWT.NONE));
+        
+        createTextLayout();
+    }
+
+    /**
+     * {@inheritDoc}
+     * @deprecated use {@link #setInput(Object)}
+     */
+    public void setInformation(final String content) {
+        setInput(new BrowserInformationControlInput(null) {
+            public String getHtml() {
+                return content;
+            }
+
+            public String getInputName() {
+                return ""; //$NON-NLS-1$
+            }
+
+            public Object getInputElement() {
+                return content;
+            }
+        });
+    }
+    
+    /**
+     * {@inheritDoc} This control can handle {@link String} and
+     * {@link BrowserInformationControlInput}.
+     */
+    public void setInput(Object input) {
+        Assert.isLegal(input is null || input instanceof String || input instanceof BrowserInformationControlInput);
+
+        if (input instanceof String) {
+            setInformation((String)input);
+            return;
+        }
+
+        fInput= (BrowserInformationControlInput)input;
+        
+        String content= null;
+        if (fInput !is null)
+            content= fInput.getHtml();
+        
+        fBrowserHasContent= content !is null && content.length() > 0;
+
+        if (!fBrowserHasContent)
+            content= "<html><body ></html>"; //$NON-NLS-1$
+
+        bool RTL= (getShell().getStyle() & DWT.RIGHT_TO_LEFT) !is 0;
+        bool resizable= isResizable();
+
+        // The default "overflow:auto" would not result in a predictable width for the client area
+        // and the re-wrapping would cause visual noise
+        String[] styles= null;
+        if (RTL && resizable)
+            styles= new String[] { "direction:rtl;", "overflow:scroll;", "word-wrap:break-word;" }; //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
+        else if (RTL && !resizable)
+            styles= new String[] { "direction:rtl;", "overflow:hidden;", "word-wrap:break-word;" }; //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
+        else if (!resizable)
+            //XXX: In IE, "word-wrap: break-word;" causes bogus wrapping even in non-broken words :-(see e.g. Javadoc of String).
+            // Re-check whether we really still need this now that the Javadoc Hover header already sets this style.
+            styles= new String[] { "overflow:hidden;"/*, "word-wrap: break-word;"*/ }; //$NON-NLS-1$
+        else
+            styles= new String[] { "overflow:scroll;" }; //$NON-NLS-1$
+        
+        StringBuffer buffer= new StringBuffer(content);
+        HTMLPrinter.insertStyles(buffer, styles);
+        content= buffer.toString();
+        
+        /*
+         * XXX: Should add some JavaScript here that shows something like
+         * "(continued...)" or "..." at the end of the visible area when the page overflowed
+         * with "overflow:hidden;".
+         */
+        
+        fCompleted= false;
+        fBrowser.setText(content);
+        
+        Object[] listeners= fInputChangeListeners.getListeners();
+        for (int i= 0; i < listeners.length; i++)
+            ((IInputChangedListener)listeners[i]).inputChanged(fInput);
+    }
+
+    /*
+     * @see IInformationControl#setVisible(bool)
+     */
+    public void setVisible(bool visible) {
+        Shell shell= getShell();
+        if (shell.isVisible() is visible)
+            return;
+        
+        if (!visible) {
+            super.setVisible(false);
+            setInput(null);
+            return;
+        }
+        
+        /*
+         * The Browser widget flickers when made visible while it is not completely loaded.
+         * The fix is to delay the call to setVisible until either loading is completed
+         * (see ProgressListener in constructor), or a timeout has been reached.
+         */
+        final Display display= shell.getDisplay();
+        
+        // Make sure the display wakes from sleep after timeout:
+        display.timerExec(100, new Runnable() {
+            public void run() {
+                fCompleted= true;
+            }
+        });
+        
+        while (!fCompleted) {
+            // Drive the event loop to process the events required to load the browser widget's contents:
+            if (!display.readAndDispatch()) {
+                display.sleep();
+            }
+        }
+        
+        shell= getShell();
+        if (shell is null || shell.isDisposed())
+            return;
+        
+        /*
+         * Avoids flickering when replacing hovers, especially on Vista in ON_CLICK mode.
+         * Causes flickering on GTK. Carbon does not care.
+         */
+        if ("win32".equals(DWT.getPlatform())) //$NON-NLS-1$
+            shell.moveAbove(null);
+        
+        super.setVisible(true);
+    }
+
+    /*
+     * @see dwtx.jface.text.AbstractInformationControl#setSize(int, int)
+     */
+    public void setSize(int width, int height) {
+        fBrowser.setRedraw(false); // avoid flickering
+        try {
+            super.setSize(width, height);
+        } finally {
+            fBrowser.setRedraw(true);
+        }
+    }
+
+    /**
+     * Creates and initializes the text layout used
+     * to compute the size hint.
+     * 
+     * @since 3.2
+     */
+    private void createTextLayout() {
+        fTextLayout= new TextLayout(fBrowser.getDisplay());
+        
+        // Initialize fonts
+        Font font= fSymbolicFontName is null ? JFaceResources.getDialogFont() : JFaceResources.getFont(fSymbolicFontName);
+        fTextLayout.setFont(font);
+        fTextLayout.setWidth(-1);
+        FontData[] fontData= font.getFontData();
+        for (int i= 0; i < fontData.length; i++)
+            fontData[i].setStyle(DWT.BOLD);
+        font= new Font(getShell().getDisplay(), fontData);
+        fBoldStyle= new TextStyle(font, null, null);
+        
+        // Compute and set tab width
+        fTextLayout.setText("    "); //$NON-NLS-1$
+        int tabWidth = fTextLayout.getBounds().width;
+        fTextLayout.setTabs(new int[] {tabWidth});
+
+        fTextLayout.setText(""); //$NON-NLS-1$
+    }
+
+    /*
+     * @see IInformationControl#dispose()
+     */
+    public void dispose() {
+        if (fTextLayout !is null) {
+            fTextLayout.dispose();
+            fTextLayout= null;
+        }
+        if (fBoldStyle !is null) {
+            fBoldStyle.font.dispose();
+            fBoldStyle= null;
+        }
+        fBrowser= null;
+        
+        super.dispose();
+    }
+
+    /*
+     * @see IInformationControl#computeSizeHint()
+     */
+    public Point computeSizeHint() {
+        Point sizeConstraints= getSizeConstraints();
+        Rectangle trim= computeTrim();
+        int height= trim.height;
+        
+        //FIXME: The HTML2TextReader does not render <p> like a browser.
+        // Instead of inserting an empty line, it just adds a single line break.
+        // Furthermore, the indentation of <dl><dd> elements is too small (e.g with a long @see line)
+        TextPresentation presentation= new TextPresentation();
+        HTML2TextReader reader= new HTML2TextReader(new StringReader(fInput.getHtml()), presentation);
+        String text;
+        try {
+            text= reader.getString();
+        } catch (IOException e) {
+            text= ""; //$NON-NLS-1$
+        }
+
+        fTextLayout.setText(text);
+        fTextLayout.setWidth(sizeConstraints is null ? DWT.DEFAULT : sizeConstraints.x - trim.width);
+        Iterator iter= presentation.getAllStyleRangeIterator();
+        while (iter.hasNext()) {
+            StyleRange sr= (StyleRange)iter.next();
+            if (sr.fontStyle is DWT.BOLD)
+                fTextLayout.setStyle(fBoldStyle, sr.start, sr.start + sr.length - 1);
+        }
+        
+        Rectangle bounds= fTextLayout.getBounds(); // does not return minimum width, see https://bugs.eclipse.org/bugs/show_bug.cgi?id=217446
+        int lineCount= fTextLayout.getLineCount();
+        int textWidth= 0;
+        for (int i= 0; i < lineCount; i++) {
+            Rectangle rect= fTextLayout.getLineBounds(i);
+            int lineWidth= rect.x + rect.width;
+            if (i is 0)
+                lineWidth += fInput.getLeadingImageWidth();
+            textWidth= Math.max(textWidth, lineWidth);
+        }
+        bounds.width= textWidth;
+        fTextLayout.setText(""); //$NON-NLS-1$
+        
+        int minWidth= bounds.width;
+        height= height + bounds.height;
+        
+        // Add some air to accommodate for different browser renderings
+        minWidth+= 15;
+        height+= 15;
+
+        
+        // Apply max size constraints
+        if (sizeConstraints !is null) {
+            if (sizeConstraints.x !is DWT.DEFAULT)
+                minWidth= Math.min(sizeConstraints.x, minWidth + trim.width);
+            if (sizeConstraints.y !is DWT.DEFAULT)
+                height= Math.min(sizeConstraints.y, height);
+        }
+
+        // Ensure minimal size
+        int width= Math.max(MIN_WIDTH, minWidth);
+        height= Math.max(MIN_HEIGHT, height);
+        
+        return new Point(width, height);
+    }
+
+    /*
+     * @see dwtx.jface.text.IInformationControlExtension3#computeTrim()
+     */
+    public Rectangle computeTrim() {
+        Rectangle trim= super.computeTrim();
+        if (isResizable()) {
+            bool RTL= (getShell().getStyle() & DWT.RIGHT_TO_LEFT) !is 0;
+            if (RTL) {
+                trim.x-= fgScrollBarSize.x;
+            }
+            trim.width+= fgScrollBarSize.x;
+            trim.height+= fgScrollBarSize.y;
+        }
+        return trim;
+    }
+    
+    /**
+     * Adds the listener to the collection of listeners who will be
+     * notified when the current location has changed or is about to change.
+     * 
+     * @param listener the location listener
+     * @since 3.4
+     */
+    public void addLocationListener(LocationListener listener) {
+        fBrowser.addLocationListener(listener);
+    }
+
+    /*
+     * @see IInformationControl#setForegroundColor(Color)
+     */
+    public void setForegroundColor(Color foreground) {
+        super.setForegroundColor(foreground);
+        fBrowser.setForeground(foreground);
+    }
+
+    /*
+     * @see IInformationControl#setBackgroundColor(Color)
+     */
+    public void setBackgroundColor(Color background) {
+        super.setBackgroundColor(background);
+        fBrowser.setBackground(background);
+    }
+
+    /*
+     * @see IInformationControlExtension#hasContents()
+     */
+    public bool hasContents() {
+        return fBrowserHasContent;
+    }
+    
+    /**
+     * Adds a listener for input changes to this input change provider.
+     * Has no effect if an identical listener is already registered.
+     * 
+     * @param inputChangeListener the listener to add
+     * @since 3.4
+     */
+    public void addInputChangeListener(IInputChangedListener inputChangeListener) {
+        Assert.isNotNull(inputChangeListener);
+        fInputChangeListeners.add(inputChangeListener);
+    }
+    
+    /**
+     * Removes the given input change listener from this input change provider.
+     * Has no effect if an identical listener is not registered.
+     * 
+     * @param inputChangeListener the listener to remove
+     * @since 3.4
+     */
+    public void removeInputChangeListener(IInputChangedListener inputChangeListener) {
+        fInputChangeListeners.remove(inputChangeListener);
+    }
+    
+    /*
+     * @see dwtx.jface.text.IDelayedInputChangeProvider#setDelayedInputChangeListener(dwtx.jface.text.IInputChangedListener)
+     * @since 3.4
+     */
+    public void setDelayedInputChangeListener(IInputChangedListener inputChangeListener) {
+        fDelayedInputChangeListener= inputChangeListener;
+    }
+    
+    /**
+     * Tells whether a delayed input change listener is registered.
+     * 
+     * @return <code>true</code> iff a delayed input change
+     *         listener is currently registered
+     * @since 3.4
+     */
+    public bool hasDelayedInputChangeListener() {
+        return fDelayedInputChangeListener !is null;
+    }
+    
+    /**
+     * Notifies listeners of a delayed input change.
+     * 
+     * @param newInput the new input, or <code>null</code> to request cancellation
+     * @since 3.4
+     */
+    public void notifyDelayedInputChange(Object newInput) {
+        if (fDelayedInputChangeListener !is null)
+            fDelayedInputChangeListener.inputChanged(newInput);
+    }
+    
+    /*
+     * @see java.lang.Object#toString()
+     * @since 3.4
+     */
+    public String toString() {
+        String style= (getShell().getStyle() & DWT.RESIZE) is 0 ? "fixed" : "resizeable"; //$NON-NLS-1$ //$NON-NLS-2$
+        return super.toString() + " -  style: " + style; //$NON-NLS-1$
+    }
+
+    /**
+     * @return the current browser input or <code>null</code>
+     */
+    public BrowserInformationControlInput getInput() {
+        return fInput;
+    }
+    
+    /*
+     * @see dwtx.jface.text.IInformationControlExtension5#computeSizeConstraints(int, int)
+     */
+    public Point computeSizeConstraints(int widthInChars, int heightInChars) {
+        if (fSymbolicFontName is null)
+            return null;
+        
+        GC gc= new GC(fBrowser);
+        Font font= fSymbolicFontName is null ? JFaceResources.getDialogFont() : JFaceResources.getFont(fSymbolicFontName);
+        gc.setFont(font);
+        int width= gc.getFontMetrics().getAverageCharWidth();
+        int height= gc.getFontMetrics().getHeight();
+        gc.dispose();
+        
+        return new Point(widthInChars * width, heightInChars * height);
+    }
+
+}