diff org.eclipse.jface/src/org/eclipse/jface/viewers/AbstractListViewer.d @ 12:bc29606a740c

Added dwt-addons in original directory structure of eclipse.org
author Frank Benoit <benoit@tionex.de>
date Sat, 14 Mar 2009 18:23:29 +0100
parents
children
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/org.eclipse.jface/src/org/eclipse/jface/viewers/AbstractListViewer.d	Sat Mar 14 18:23:29 2009 +0100
@@ -0,0 +1,566 @@
+/*******************************************************************************
+ * Copyright (c) 2004, 2007 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
+ *     Chris Longfield <clongfield@internap.com> - Fix for Bug 70856
+ *     Tom Schindl - fix for bug 157309
+ *     Brad Reynolds - bug 141435
+ * Port to the D programming language:
+ *     Frank Benoit <benoit@tionex.de>
+ *******************************************************************************/
+
+module org.eclipse.jface.viewers.AbstractListViewer;
+
+import org.eclipse.jface.viewers.StructuredViewer;
+import org.eclipse.jface.viewers.ILabelProvider;
+import org.eclipse.jface.viewers.IBaseLabelProvider;
+import org.eclipse.jface.viewers.ViewerComparator;
+import org.eclipse.jface.viewers.IElementComparer;
+
+
+import org.eclipse.swt.widgets.Control;
+import org.eclipse.swt.widgets.Widget;
+import org.eclipse.core.runtime.Assert;
+
+import java.lang.all;
+import java.util.Collection;
+import java.util.List;
+import java.util.ArrayList;
+import java.util.Set;
+
+/**
+ * Abstract base class for viewers that contain lists of items (such as a combo or list).
+ * Most of the viewer implementation is in this base class, except for the minimal code that
+ * actually communicates with the underlying widget.
+ *
+ * @see org.eclipse.jface.viewers.ListViewer
+ * @see org.eclipse.jface.viewers.ComboViewer
+ *
+ * @since 3.0
+ */
+public abstract class AbstractListViewer : StructuredViewer {
+    alias StructuredViewer.internalRefresh internalRefresh;
+    alias StructuredViewer.setSelectionToWidget setSelectionToWidget;
+
+    /**
+     * A list of viewer elements (element type: <code>Object</code>).
+     */
+    private List listMap;
+
+    /**
+     * Adds the given string to the underlying widget at the given index
+     *
+     * @param string the string to add
+     * @param index position to insert the string into
+     */
+    protected abstract void listAdd(String string, int index);
+
+    /**
+     * Sets the text of the item at the given index in the underlying widget.
+     *
+     * @param index index to modify
+     * @param string new text
+     */
+    protected abstract void listSetItem(int index, String string);
+
+    /**
+     * Returns the zero-relative indices of the items which are currently
+     * selected in the underlying widget.  The array is empty if no items are selected.
+     * <p>
+     * Note: This is not the actual structure used by the receiver
+     * to maintain its selection, so modifying the array will
+     * not affect the receiver.
+     * </p>
+     * @return the array of indices of the selected items
+     */
+    protected abstract int[] listGetSelectionIndices();
+
+    /**
+     * Returns the number of items contained in the underlying widget.
+     *
+     * @return the number of items
+     */
+    protected abstract int listGetItemCount();
+
+    /**
+     * Sets the underlying widget's items to be the given array of items.
+     *
+     * @param labels the array of label text
+     */
+    protected abstract void listSetItems(String[] labels);
+
+    /**
+     * Removes all of the items from the underlying widget.
+     */
+    protected abstract void listRemoveAll();
+
+    /**
+     * Removes the item from the underlying widget at the given
+     * zero-relative index.
+     *
+     * @param index the index for the item
+     */
+    protected abstract void listRemove(int index);
+
+    /**
+     * Selects the items at the given zero-relative indices in the underlying widget.
+     * The current selection is cleared before the new items are selected.
+     * <p>
+     * Indices that are out of range and duplicate indices are ignored.
+     * If the receiver is single-select and multiple indices are specified,
+     * then all indices are ignored.
+     *
+     * @param ixs the indices of the items to select
+     */
+    protected abstract void listSetSelection(int[] ixs);
+
+    /**
+     * Shows the selection.  If the selection is already showing in the receiver,
+     * this method simply returns.  Otherwise, the items are scrolled until
+     * the selection is visible.
+     */
+    protected abstract void listShowSelection();
+
+    /**
+     * Deselects all selected items in the underlying widget.
+     */
+    protected abstract void listDeselectAll();
+
+    public this(){
+        listMap = new ArrayList();
+    }
+
+    /**
+     * Adds the given elements to this list viewer.
+     * If this viewer does not have a sorter, the elements are added at the end
+     * in the order given; otherwise the elements are inserted at appropriate positions.
+     * <p>
+     * This method should be called (by the content provider) when elements
+     * have been added to the model, in order to cause the viewer to accurately
+     * reflect the model. This method only affects the viewer, not the model.
+     * </p>
+     *
+     * @param elements the elements to add
+     */
+    public void add(Object[] elements) {
+        assertElementsNotNull(elements);
+        Object[] filtered = filter(elements);
+        ILabelProvider labelProvider = cast(ILabelProvider) getLabelProvider();
+        for (int i = 0; i < filtered.length; i++) {
+            Object element = filtered[i];
+            int ix = indexForElement(element);
+            insertItem(labelProvider, element, ix);
+        }
+    }
+
+    private void insertItem(ILabelProvider labelProvider, Object element, int index) {
+        listAdd(getLabelProviderText(labelProvider, element), index);
+        listMap.add(index, element);
+        mapElement(element, getControl()); // must map it, since findItem only looks in map, if enabled
+    }
+
+    /**
+     * Inserts the given element into this list viewer at the given position.
+     * If this viewer has a sorter, the position is ignored and the element is
+     * inserted at the correct position in the sort order.
+     * <p>
+     * This method should be called (by the content provider) when elements have
+     * been added to the model, in order to cause the viewer to accurately
+     * reflect the model. This method only affects the viewer, not the model.
+     * </p>
+     *
+     * @param element
+     *            the element
+     * @param position
+     *            a 0-based position relative to the model, or -1 to indicate
+     *            the last position
+     * @since 3.3
+     */
+    public void insert(Object element, int position) {
+        if (getComparator() !is null || hasFilters()) {
+            add(element);
+            return;
+        }
+
+        insertItem(cast(ILabelProvider) getLabelProvider(), element, position);
+    }
+
+
+    /**
+     * Return the text for the element from the labelProvider.
+     * If it is null then return the empty String.
+     * @param labelProvider ILabelProvider
+     * @param element
+     * @return String. Return the emptyString if the labelProvider
+     * returns null for the text.
+     *
+     * @since 3.1
+     */
+    private String getLabelProviderText(ILabelProvider labelProvider, Object element){
+        String text = labelProvider.getText(element);
+        if(text is null) {
+            return "";//$NON-NLS-1$
+        }
+        return text;
+    }
+
+    /**
+     * Adds the given element to this list viewer.
+     * If this viewer does not have a sorter, the element is added at the end;
+     * otherwise the element is inserted at the appropriate position.
+     * <p>
+     * This method should be called (by the content provider) when a single element
+     * has been added to the model, in order to cause the viewer to accurately
+     * reflect the model. This method only affects the viewer, not the model.
+     * Note that there is another method for efficiently processing the simultaneous
+     * addition of multiple elements.
+     * </p>
+     *
+     * @param element the element
+     */
+    public void add(Object element) {
+        add([ element ]);
+    }
+
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     * Since SWT.List doesn't use items we always return the List itself.
+     */
+    protected override Widget doFindInputItem(Object element) {
+        if (element !is null && opEquals(element, getRoot())) {
+            return getControl();
+        }
+        return null;
+    }
+
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     * Since SWT.List doesn't use items we always return the List itself.
+     */
+    protected override Widget doFindItem(Object element) {
+        if (element !is null) {
+            if (listMapContains(element)) {
+                return getControl();
+            }
+        }
+        return null;
+    }
+
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     */
+    protected override void doUpdateItem(Widget data, Object element, bool fullMap) {
+        if (element !is null) {
+            int ix = getElementIndex(element);
+            if (ix >= 0) {
+                ILabelProvider labelProvider = cast(ILabelProvider) getLabelProvider();
+                listSetItem(ix, getLabelProviderText(labelProvider,element));
+            }
+        }
+    }
+
+    /**
+     * Returns the element with the given index from this list viewer.
+     * Returns <code>null</code> if the index is out of range.
+     *
+     * @param index the zero-based index
+     * @return the element at the given index, or <code>null</code> if the
+     *   index is out of range
+     */
+    public Object getElementAt(int index) {
+        if (index >= 0 && index < listMap.size()) {
+            return listMap.get(index);
+        }
+        return null;
+    }
+
+    /**
+     * The list viewer implementation of this <code>Viewer</code> framework
+     * method returns the label provider, which in the case of list
+     * viewers will be an instance of <code>ILabelProvider</code>.
+     */
+    public override IBaseLabelProvider getLabelProvider() {
+        return super.getLabelProvider();
+    }
+
+    /* (non-Javadoc)
+     * Method declared on Viewer.
+     */
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     */
+    protected override List getSelectionFromWidget() {
+        int[] ixs = listGetSelectionIndices();
+        ArrayList list = new ArrayList(ixs.length);
+        for (int i = 0; i < ixs.length; i++) {
+            Object e = getElementAt(ixs[i]);
+            if (e !is null) {
+                list.add(e);
+            }
+        }
+        return list;
+    }
+
+    /**
+     * @param element the element to insert
+     * @return the index where the item should be inserted.
+     */
+    protected int indexForElement(Object element) {
+        ViewerComparator comparator = getComparator();
+        if (comparator is null) {
+            return listGetItemCount();
+        }
+        int count = listGetItemCount();
+        int min = 0, max = count - 1;
+        while (min <= max) {
+            int mid = (min + max) / 2;
+            Object data = listMap.get(mid);
+            int compare = comparator.compare(this, data, element);
+            if (compare is 0) {
+                // find first item > element
+                while (compare is 0) {
+                    ++mid;
+                    if (mid >= count) {
+                        break;
+                    }
+                    data = listMap.get(mid);
+                    compare = comparator.compare(this, data, element);
+                }
+                return mid;
+            }
+            if (compare < 0) {
+                min = mid + 1;
+            } else {
+                max = mid - 1;
+            }
+        }
+        return min;
+    }
+
+    /* (non-Javadoc)
+     * Method declared on Viewer.
+     */
+    protected override void inputChanged(Object input, Object oldInput) {
+        listMap.clear();
+        Object[] children = getSortedChildren(getRoot());
+        int size = children.length;
+
+        listRemoveAll();
+        String[] labels = new String[size];
+        for (int i = 0; i < size; i++) {
+            Object el = children[i];
+            labels[i] = getLabelProviderText(cast(ILabelProvider) getLabelProvider(),el);
+            listMap.add(el);
+            mapElement(el, getControl()); // must map it, since findItem only looks in map, if enabled
+        }
+        listSetItems(labels);
+    }
+
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     */
+    protected override void internalRefresh(Object element) {
+        Control list = getControl();
+        if (element is null || opEquals(element, getRoot())) {
+            // the parent
+            if (listMap !is null) {
+                listMap.clear();
+            }
+            unmapAllElements();
+            List selection = getSelectionFromWidget();
+
+            int topIndex = -1;
+            if (selection is null || selection.isEmpty()) {
+                topIndex = listGetTopIndex();
+            }
+
+            Object[] children = null;
+            list.setRedraw(false);
+            try {
+                listRemoveAll();
+
+                children = getSortedChildren(getRoot());
+                String[] items = new String[children.length];
+
+                ILabelProvider labelProvider = cast(ILabelProvider) getLabelProvider();
+
+                for (int i = 0; i < items.length; i++) {
+                    Object el = children[i];
+                    items[i] = getLabelProviderText(labelProvider, el);
+                    listMap.add(el);
+                    mapElement(el, list); // must map it, since findItem only looks in map, if enabled
+                }
+
+                listSetItems(items);
+            } finally {
+                list.setRedraw(true);
+            }
+
+            if (topIndex is -1) {
+                setSelectionToWidget(selection, false);
+            } else {
+                listSetTopIndex(Math.min(topIndex, children.length));
+            }
+        } else {
+            doUpdateItem(list, element, true);
+        }
+    }
+
+    /**
+     * Returns the index of the item currently at the top of the viewable area.
+     * <p>
+     * Default implementation returns -1.
+     * </p>
+     * @return index, -1 for none
+     * @since 3.3
+     */
+    protected int listGetTopIndex(){
+        return -1;
+    }
+
+    /**
+     * Sets the index of the item to be at the top of the viewable area.
+     * <p>
+     * Default implementation does nothing.
+     * </p>
+     * @param index the given index. -1 for none.  index will always refer to a valid index.
+     * @since 3.3
+     */
+    protected void listSetTopIndex(int index) {
+    }
+
+    /**
+     * Removes the given elements from this list viewer.
+     *
+     * @param elements the elements to remove
+     */
+    private void internalRemove(Object[] elements) {
+        Object input = getInput();
+        for (int i = 0; i < elements.length; ++i) {
+            if ( opEquals(elements[i], input)) {
+                setInput(null);
+                return;
+            }
+            int ix = getElementIndex(elements[i]);
+            if (ix >= 0) {
+                listRemove(ix);
+                listMap.remove(ix);
+                unmapElement(elements[i], getControl());
+            }
+        }
+    }
+
+    /**
+     * Removes the given elements from this list viewer.
+     * The selection is updated if required.
+     * <p>
+     * This method should be called (by the content provider) when elements
+     * have been removed from the model, in order to cause the viewer to accurately
+     * reflect the model. This method only affects the viewer, not the model.
+     * </p>
+     *
+     * @param elements the elements to remove
+     */
+    public void remove(Object[] elements) {
+        assertElementsNotNull(elements);
+        if (elements.length is 0) {
+            return;
+        }
+        preservingSelection( dgRunnable( (Object[] elements_){
+            internalRemove(elements_);
+        }, elements));
+    }
+
+    /**
+     * Removes the given element from this list viewer.
+     * The selection is updated if necessary.
+     * <p>
+     * This method should be called (by the content provider) when a single element
+     * has been removed from the model, in order to cause the viewer to accurately
+     * reflect the model. This method only affects the viewer, not the model.
+     * Note that there is another method for efficiently processing the simultaneous
+     * removal of multiple elements.
+     * </p>
+     *
+     * @param element the element
+     */
+    public void remove(Object element) {
+        remove([ element ]);
+    }
+
+    /**
+     * The list viewer implementation of this <code>Viewer</code> framework
+     * method ensures that the given label provider is an instance of
+     * <code>ILabelProvider</code>.
+     *
+     * <b>The optional interfaces {@link IColorProvider} and
+     * {@link IFontProvider} have no effect for this type of viewer</b>
+     */
+    public override void setLabelProvider(IBaseLabelProvider labelProvider) {
+        Assert.isTrue( null !is cast(ILabelProvider)labelProvider );
+        super.setLabelProvider(labelProvider);
+    }
+
+    /* (non-Javadoc)
+     * Method declared on StructuredViewer.
+     */
+    protected override void setSelectionToWidget(List in_, bool reveal) {
+        if (in_ is null || in_.size() is 0) { // clear selection
+            listDeselectAll();
+        } else {
+            int n = in_.size();
+            int[] ixs = new int[n];
+            int count = 0;
+            for (int i = 0; i < n; ++i) {
+                Object el = in_.get(i);
+                int ix = getElementIndex(el);
+                if (ix >= 0) {
+                    ixs[count++] = ix;
+                }
+            }
+            if (count < n) {
+                System.arraycopy(ixs, 0, ixs = new int[count], 0, count);
+            }
+            listSetSelection(ixs);
+            if (reveal) {
+                listShowSelection();
+            }
+        }
+    }
+
+    /**
+     * Returns the index of the given element in listMap, or -1 if the element cannot be found.
+     * As of 3.3, uses the element comparer if available.
+     *
+     * @param element
+     * @return the index
+     */
+    int getElementIndex(Object element) {
+        IElementComparer comparer = getComparer();
+        if (comparer is null) {
+            return listMap.indexOf(element);
+        }
+        int size = listMap.size();
+        for (int i = 0; i < size; i++) {
+            if (comparer.opEquals(element, listMap.get(i)))
+                return i;
+        }
+        return -1;
+    }
+
+    /**
+     * @param element
+     * @return true if listMap contains the given element
+     *
+     * @since 3.3
+     */
+    private bool listMapContains(Object element) {
+        return getElementIndex(element) !is -1;
+    }
+
+}