net.enilink.commons.ui.jface.table.CComboBoxCellEditor.java Source code

Java tutorial

Introduction

Here is the source code for net.enilink.commons.ui.jface.table.CComboBoxCellEditor.java

Source

/*******************************************************************************
 * Copyright (c) 2000, 2009 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
 *     Tom Schindl <tom.schindl@bestsolution.at> - bugfix in 174739
 *     Ken Wenzel - adaption to CCombo
 *******************************************************************************/

package net.enilink.commons.ui.jface.table;

import java.text.MessageFormat;

import org.eclipse.core.runtime.Assert;
import org.eclipse.jface.viewers.CellEditor;
import org.eclipse.jface.viewers.IBaseLabelProvider;
import org.eclipse.jface.viewers.IContentProvider;
import org.eclipse.jface.viewers.ILabelProvider;
import org.eclipse.jface.viewers.IStructuredContentProvider;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.jface.viewers.StructuredSelection;
import org.eclipse.jface.viewers.Viewer;
import org.eclipse.jface.viewers.ViewerComparator;
import org.eclipse.swt.SWT;
import org.eclipse.swt.custom.CCombo;
import org.eclipse.swt.events.FocusAdapter;
import org.eclipse.swt.events.FocusEvent;
import org.eclipse.swt.events.FocusListener;
import org.eclipse.swt.events.KeyAdapter;
import org.eclipse.swt.events.KeyEvent;
import org.eclipse.swt.events.SelectionAdapter;
import org.eclipse.swt.events.SelectionEvent;
import org.eclipse.swt.events.TraverseEvent;
import org.eclipse.swt.events.TraverseListener;
import org.eclipse.swt.graphics.GC;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;

import net.enilink.commons.ui.jface.viewers.CComboViewer;

/**
 * A cell editor that presents a list of items in a combo box.
 */
public class CComboBoxCellEditor extends CellEditor {
    private IBaseLabelProvider labelProvider;

    private IContentProvider contentProvider;

    private ViewerComparator viewerComparator;

    /**
     * The list of items to present in the combo box.
     */
    private Object[] items;

    /**
     * The the selected item.
     */
    protected Object selection;

    /**
     * The custom combo box control.
     */
    protected CCombo comboBox;

    /**
     * The combo box viewer
     */
    protected CComboViewer viewer;

    private Object input;

    /**
     * Default ComboBoxCellEditor style
     */
    private static final int defaultStyle = SWT.NONE;

    /**
     * Creates a new cell editor with no control and no st of choices.
     * Initially, the cell editor has no cell validator.
     * 
     * @see CellEditor#setStyle
     * @see CellEditor#create
     * @see CComboBoxCellEditor#setItems
     * @see CellEditor#dispose
     */
    public CComboBoxCellEditor() {
        setStyle(defaultStyle);
    }

    /**
     * Creates a new cell editor with a combo containing the given list of
     * choices and parented under the given control. The cell editor value is
     * the zero-based index of the selected item. Initially, the cell editor has
     * no cell validator and the first item in the list is selected.
     * 
     * @param parent
     *            the parent control
     * @param items
     *            the list of strings for the combo box
     * @param labelProvider
     *            the label provider of the comboviewer
     * @param style
     *            the style bits
     */
    public CComboBoxCellEditor(Composite parent, Object[] items, IBaseLabelProvider labelProvider, int style) {
        super(parent, style);
        this.labelProvider = labelProvider;
        setItems(items);
    }

    /**
     * Creates a new cell editor with a combo containing the given list of
     * choices and parented under the given control. The cell editor value is
     * the zero-based index of the selected item. Initially, the cell editor has
     * no cell validator and the first item in the list is selected.
     * 
     * @param parent
     *            the parent control
     * @param items
     *            the list of strings for the combo box
     * @param style
     *            the style bits
     */
    public CComboBoxCellEditor(Composite parent, ILabelProvider labelProvider, IContentProvider contentProvider,
            int style) {
        setStyle(style);
        this.labelProvider = labelProvider;
        this.contentProvider = contentProvider;
        create(parent);
    }

    /**
     * Returns the list of choices for the combo box
     * 
     * @return the list of choices for the combo box
     */
    public Object[] getItems() {
        return this.items;
    }

    /**
     * Sets the list of choices for the combo box
     * 
     * @param items
     *            the list of choices for the combo box
     */
    public void setItems(Object[] items) {
        Assert.isNotNull(items);
        this.items = items;
        populateComboBoxItems();
    }

    /**
     * Sets the list of choices for the combo box
     * 
     * @param items
     *            the list of choices for the combo box
     */
    public void setInput(Object input) {
        this.input = input;
        if (viewer != null) {
            viewer.setInput(input);

            setValueValid(true);
            selection = null;
        }
    }

    protected FocusListener createFocusListener() {
        return new FocusAdapter() {
            public void focusLost(FocusEvent e) {
                CComboBoxCellEditor.this.focusLost();
            }
        };
    }

    /*
     * (non-Javadoc) Method declared on CellEditor.
     */
    protected Control createControl(Composite parent) {
        comboBox = new CCombo(parent, getStyle());
        comboBox.setFont(parent.getFont());

        comboBox.addKeyListener(new KeyAdapter() {
            // hook key pressed - see PR 14201
            public void keyPressed(KeyEvent e) {
                keyReleaseOccured(e);
            }
        });

        comboBox.addSelectionListener(new SelectionAdapter() {
            public void widgetDefaultSelected(SelectionEvent event) {
                applyEditorValueAndDeactivate();
            }

            public void widgetSelected(SelectionEvent event) {
                selection = ((IStructuredSelection) viewer.getSelection()).getFirstElement();
            }
        });

        comboBox.addTraverseListener(new TraverseListener() {
            public void keyTraversed(TraverseEvent e) {
                if (e.detail == SWT.TRAVERSE_ESCAPE || e.detail == SWT.TRAVERSE_RETURN) {
                    e.doit = false;
                }
            }
        });

        FocusListener focusListener = createFocusListener();
        if (focusListener != null) {
            comboBox.addFocusListener(focusListener);
        }

        viewer = new CComboViewer(comboBox);

        if (contentProvider == null) {
            contentProvider = new IStructuredContentProvider() {
                public void dispose() {

                }

                public void inputChanged(Viewer viewer, Object oldInput, Object newInput) {

                }

                public Object[] getElements(Object inputElement) {
                    if (inputElement instanceof Object[]) {
                        return (Object[]) inputElement;
                    }
                    return new Object[0];
                }
            };
        }
        viewer.setContentProvider(contentProvider);
        if (labelProvider != null) {
            viewer.setLabelProvider(labelProvider);
        }

        if (viewerComparator != null) {
            viewer.setComparator(viewerComparator);
        }

        return comboBox;
    }

    protected void selectionChanged() {

    }

    public void setLabelProvider(IBaseLabelProvider labelProvider) {
        this.labelProvider = labelProvider;
        if (viewer != null && labelProvider != null) {
            viewer.setLabelProvider(labelProvider);
        }
    }

    public void setContentProvider(IContentProvider contentProvider) {
        this.contentProvider = contentProvider;
        if (viewer != null && contentProvider != null) {
            viewer.setContentProvider(contentProvider);
        }
    }

    public void setViewerComparator(ViewerComparator viewerComparator) {
        this.viewerComparator = viewerComparator;
        if (viewer != null && viewerComparator != null) {
            viewer.setComparator(viewerComparator);
        }
    }

    /**
     * The <code>ComboBoxCellEditor</code> implementation of this
     * <code>CellEditor</code> framework method returns the current selection.
     * 
     * @return the current selection
     */
    protected Object doGetValue() {
        return selection;
    }

    /*
     * (non-Javadoc) Method declared on CellEditor.
     */
    protected void doSetFocus() {
        comboBox.setFocus();
    }

    /**
     * The <code>ComboBoxCellEditor</code> implementation of this
     * <code>CellEditor</code> framework method sets the minimum width of the
     * cell. The minimum width is 10 characters if <code>comboBox</code> is
     * not <code>null</code> or <code>disposed</code> eles it is 60 pixels
     * to make sure the arrow button and some text is visible. The list of
     * CCombo will be wide enough to show its longest item.
     */
    public LayoutData getLayoutData() {
        LayoutData layoutData = super.getLayoutData();
        if (comboBox == null || comboBox.isDisposed()) {
            layoutData.minimumWidth = 60;
        } else {
            // make the comboBox 10 characters wide
            GC gc = new GC(comboBox);
            layoutData.minimumWidth = (gc.getFontMetrics().getAverageCharWidth() * 10) + 10;
            gc.dispose();
        }
        return layoutData;
    }

    /**
     * The <code>ComboBoxCellEditor</code> implementation of this
     * <code>CellEditor</code> framework method accepts a zero-based index of
     * a selection.
     * 
     * @param value
     *            the zero-based index of the selection wrapped as an
     *            <code>Integer</code>
     */
    protected void doSetValue(Object value) {
        Assert.isTrue(viewer != null);
        if (value != null) {
            viewer.setSelection(new StructuredSelection(value), true);
        } else {
            viewer.setSelection(StructuredSelection.EMPTY);
        }
        selection = value;
    }

    /**
     * Updates the list of choices for the combo box for the current control.
     */
    private void populateComboBoxItems() {
        if (viewer != null && items != null) {
            viewer.setInput(items);

            setValueValid(true);
            selection = null;
        }
    }

    /**
     * Applies the currently selected value and deactiavates the cell editor
     */
    void applyEditorValueAndDeactivate() {
        // must set the selection before getting value
        selection = ((IStructuredSelection) viewer.getSelection()).getFirstElement();
        Object newValue = doGetValue();
        markDirty();
        boolean isValid = isCorrect(newValue);
        setValueValid(isValid);

        if (!isValid) {
            // Only format if the selection is valid
            if (selection != null) {
                // try to insert the current value into the error message.
                setErrorMessage(MessageFormat.format(getErrorMessage(), new Object[] { selection }));
            } else {
                // Since we don't have a valid selection, assume we're using an
                // 'edit'
                // combo so format using its text value
                setErrorMessage(MessageFormat.format(getErrorMessage(), new Object[] { comboBox.getText() }));
            }
        }

        fireApplyEditorValue();
        deactivate();
    }

    /*
     * (non-Javadoc)
     * 
     * @see org.eclipse.jface.viewers.CellEditor#focusLost()
     */
    protected void focusLost() {
        if (isActivated()) {
            applyEditorValueAndDeactivate();
        }
    }

    /*
     * (non-Javadoc)
     * 
     * @see org.eclipse.jface.viewers.CellEditor#keyReleaseOccured(org.eclipse.swt.events.KeyEvent)
     */
    protected void keyReleaseOccured(KeyEvent keyEvent) {
        if (keyEvent.character == '\u001b') { // Escape character
            fireCancelEditor();
        } else if (keyEvent.character == '\t') { // tab key
            applyEditorValueAndDeactivate();
        }
    }
}