在TableViewer或TreeViewer编辑时候,Eclipse提供了基本的CellEditor,如TextCellEditor、CheckboxCellEditor、ComboBoxCellEditor、DialogCellEditor等,但在实际应用过程中,我们通常有特殊需要,如下图类型的单元格编辑器:

实现的方式相当简单,我组合了ComboBoxCellEditor、DialogCellEditor,其中ComboBox支持手工输入,返回为String类型值(原ComboBoxCellEditor为Integer类型),贴下具体代码吧:
 package com.test.ui.properties.invoke;
package com.test.ui.properties.invoke;

 import java.text.MessageFormat;
import java.text.MessageFormat;

 import org.eclipse.core.runtime.Assert;
import org.eclipse.core.runtime.Assert;
 import org.eclipse.jface.viewers.CellEditor;
import org.eclipse.jface.viewers.CellEditor;
 import org.eclipse.swt.SWT;
import org.eclipse.swt.SWT;
 import org.eclipse.swt.custom.CCombo;
import org.eclipse.swt.custom.CCombo;
 import org.eclipse.swt.events.FocusAdapter;
import org.eclipse.swt.events.FocusAdapter;
 import org.eclipse.swt.events.FocusEvent;
import org.eclipse.swt.events.FocusEvent;
 import org.eclipse.swt.events.FocusListener;
import org.eclipse.swt.events.FocusListener;
 import org.eclipse.swt.events.KeyAdapter;
import org.eclipse.swt.events.KeyAdapter;
 import org.eclipse.swt.events.KeyEvent;
import org.eclipse.swt.events.KeyEvent;
 import org.eclipse.swt.events.ModifyEvent;
import org.eclipse.swt.events.ModifyEvent;
 import org.eclipse.swt.events.ModifyListener;
import org.eclipse.swt.events.ModifyListener;
 import org.eclipse.swt.events.SelectionAdapter;
import org.eclipse.swt.events.SelectionAdapter;
 import org.eclipse.swt.events.SelectionEvent;
import org.eclipse.swt.events.SelectionEvent;
 import org.eclipse.swt.events.TraverseEvent;
import org.eclipse.swt.events.TraverseEvent;
 import org.eclipse.swt.events.TraverseListener;
import org.eclipse.swt.events.TraverseListener;
 import org.eclipse.swt.graphics.Color;
import org.eclipse.swt.graphics.Color;
 import org.eclipse.swt.graphics.Font;
import org.eclipse.swt.graphics.Font;
 import org.eclipse.swt.graphics.Point;
import org.eclipse.swt.graphics.Point;
 import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.graphics.Rectangle;
 import org.eclipse.swt.widgets.Button;
import org.eclipse.swt.widgets.Button;
 import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Composite;
 import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Control;
 import org.eclipse.swt.widgets.Layout;
import org.eclipse.swt.widgets.Layout;


 public abstract class ComboBoxDialogCellEditor extends CellEditor
public abstract class ComboBoxDialogCellEditor extends CellEditor  {
{

 //Combo Items
    //Combo Items
 private String[] items;
    private String[] items;

 private Composite editor;
    private Composite editor;
 
    
 private CCombo comboBox;
    private CCombo comboBox;
 
    
 private Control contents;
    private Control contents;
 
    
 private Button button;
    private Button button;
 private FocusListener buttonFocusListener;
    private FocusListener buttonFocusListener;

 private ModifyListener modifyListener;
    private ModifyListener modifyListener;
 
    
 private Object value = null;
    private Object value = null;


 /** *//**
    /** *//**
 * Internal class for laying out the dialog.
     * Internal class for laying out the dialog.
 */
     */

 private class DialogCellLayout extends Layout
    private class DialogCellLayout extends Layout  {
{

 public void layout(Composite editor, boolean force)
        public void layout(Composite editor, boolean force)  {
{
 Rectangle bounds = editor.getClientArea();
            Rectangle bounds = editor.getClientArea();
 Point size = button.computeSize(SWT.DEFAULT, SWT.DEFAULT, force);
            Point size = button.computeSize(SWT.DEFAULT, SWT.DEFAULT, force);

 if (contents != null)
            if (contents != null)  {
{
 contents.setBounds(0, 0, bounds.width - size.x, bounds.height);
                contents.setBounds(0, 0, bounds.width - size.x, bounds.height);
 }
            }
 button.setBounds(bounds.width - size.x, 0, size.x, bounds.height);
            button.setBounds(bounds.width - size.x, 0, size.x, bounds.height);
 }
        }

 public Point computeSize(Composite editor, int wHint, int hHint,
        public Point computeSize(Composite editor, int wHint, int hHint,

 boolean force)
                boolean force)  {
{

 if (wHint != SWT.DEFAULT && hHint != SWT.DEFAULT)
            if (wHint != SWT.DEFAULT && hHint != SWT.DEFAULT)  {
{
 return new Point(wHint, hHint);
                return new Point(wHint, hHint);
 }
            }
 Point contentsSize = contents.computeSize(SWT.DEFAULT, SWT.DEFAULT,
            Point contentsSize = contents.computeSize(SWT.DEFAULT, SWT.DEFAULT,
 force);
                    force);
 Point buttonSize = button.computeSize(SWT.DEFAULT, SWT.DEFAULT,
            Point buttonSize = button.computeSize(SWT.DEFAULT, SWT.DEFAULT,
 force);
                    force);
 // Just return the button width to ensure the button is not clipped
            // Just return the button width to ensure the button is not clipped
 // if the label is long.
            // if the label is long.
 // The label will just use whatever extra width there is
            // The label will just use whatever extra width there is
 Point result = new Point(buttonSize.x, Math.max(contentsSize.y,
            Point result = new Point(buttonSize.x, Math.max(contentsSize.y,
 buttonSize.y));
                    buttonSize.y));
 return result;
            return result;
 }
        }
 }
    }

 //Combo default style
    //Combo default style
 private static final int defaultStyle = SWT.NONE;
    private static final int defaultStyle = SWT.NONE;

 
    

 public ComboBoxDialogCellEditor()
    public ComboBoxDialogCellEditor()  {
{
 setStyle(defaultStyle);
        setStyle(defaultStyle);
 }
    }


 public ComboBoxDialogCellEditor(Composite parent, String[] items)
    public ComboBoxDialogCellEditor(Composite parent, String[] items)  {
{
 this(parent, items, defaultStyle);
        this(parent, items, defaultStyle);
 }
    }


 public ComboBoxDialogCellEditor(Composite parent, String[] items, int style)
    public ComboBoxDialogCellEditor(Composite parent, String[] items, int style)  {
{
 super(parent, style);
        super(parent, style);
 setItems(items);
        setItems(items);
 }
    }


 public String[] getItems()
    public String[] getItems()  {
{
 return items;
        return items;
 }
    }


 public void setItems(String[] items)
    public void setItems(String[] items)  {
{
 Assert.isNotNull(items);
        Assert.isNotNull(items);
 this.items = items;
        this.items = items;
 populateComboBoxItems();
        populateComboBoxItems();
 }
    }
 
    

 protected Button createButton(Composite parent)
    protected Button createButton(Composite parent)  {
{
 Button result = new Button(parent, SWT.DOWN);
        Button result = new Button(parent, SWT.DOWN);
 result.setText("
        result.setText(" "); //$NON-NLS-1$
"); //$NON-NLS-1$
 return result;
        return result;
 }
    }


 protected Control createContents(Composite cell)
    protected Control createContents(Composite cell)  {
{
 comboBox = new CCombo(cell, getStyle());
        comboBox = new CCombo(cell, getStyle());
 comboBox.setFont(cell.getFont());
        comboBox.setFont(cell.getFont());
 populateComboBoxItems();
        populateComboBoxItems();


 comboBox.addKeyListener(new KeyAdapter()
        comboBox.addKeyListener(new KeyAdapter()  {
{

 public void keyPressed(KeyEvent e)
            public void keyPressed(KeyEvent e)  {
{
 keyReleaseOccured(e);
                keyReleaseOccured(e);
 }
            }
 });
        });

 comboBox.addModifyListener(getModifyListener());
        comboBox.addModifyListener(getModifyListener());


 comboBox.addSelectionListener(new SelectionAdapter()
        comboBox.addSelectionListener(new SelectionAdapter()  {
{

 public void widgetDefaultSelected(SelectionEvent event)
            public void widgetDefaultSelected(SelectionEvent event)  {
{
 applyEditorValueAndDeactivate();
                applyEditorValueAndDeactivate();
 }
            }
 
            

 public void widgetSelected(SelectionEvent event)
            public void widgetSelected(SelectionEvent event)  {
{
 value = comboBox.getText();
                value = comboBox.getText();
 }
            }
 });
        });
 
        

 comboBox.addTraverseListener(new TraverseListener()
        comboBox.addTraverseListener(new TraverseListener()  {
{

 public void keyTraversed(TraverseEvent e)
            public void keyTraversed(TraverseEvent e)  {
{
 if (e.detail == SWT.TRAVERSE_ESCAPE
                if (e.detail == SWT.TRAVERSE_ESCAPE

 || e.detail == SWT.TRAVERSE_RETURN)
                        || e.detail == SWT.TRAVERSE_RETURN)  {
{
 e.doit = false;
                    e.doit = false;
 }
                }
 }
            }
 });
        });


 comboBox.addFocusListener(new FocusAdapter()
        comboBox.addFocusListener(new FocusAdapter()  {
{

 public void focusLost(FocusEvent e)
            public void focusLost(FocusEvent e)  {
{

 if(!button.isFocusControl())
                if(!button.isFocusControl())  {
{
 ComboBoxDialogCellEditor.this.focusLost();
                    ComboBoxDialogCellEditor.this.focusLost();
 }
                }
 }
            }
 });
        });

 return comboBox;
        return comboBox;
 }
    }
 
    
 @Override
    @Override

 protected Control createControl(Composite parent)
    protected Control createControl(Composite parent)  {
{
 Font font = parent.getFont();
        Font font = parent.getFont();
 Color bg = parent.getBackground();
        Color bg = parent.getBackground();

 editor = new Composite(parent, getStyle());
        editor = new Composite(parent, getStyle());
 editor.setFont(font);
        editor.setFont(font);
 editor.setBackground(bg);
        editor.setBackground(bg);
 editor.setLayout(new DialogCellLayout());
        editor.setLayout(new DialogCellLayout());

 contents = createContents(editor);
        contents = createContents(editor);
 updateContents(value);
        updateContents(value);
 
        
 button = createButton(editor);
        button = createButton(editor);
 button.setFont(font);
        button.setFont(font);


 button.addKeyListener(new KeyAdapter()
        button.addKeyListener(new KeyAdapter()  {
{

 public void keyReleased(KeyEvent e)
            public void keyReleased(KeyEvent e)  {
{

 if (e.character == '\u001b')
                if (e.character == '\u001b')  { // Escape
{ // Escape
 fireCancelEditor();
                    fireCancelEditor();
 }
                }
 }
            }
 });
        });
 button.addFocusListener(getButtonFocusListener());
        button.addFocusListener(getButtonFocusListener());

 button.addSelectionListener(new SelectionAdapter()
        button.addSelectionListener(new SelectionAdapter()  {
{

 public void widgetSelected(SelectionEvent event)
            public void widgetSelected(SelectionEvent event)  {
{
 button.removeFocusListener(getButtonFocusListener());
                button.removeFocusListener(getButtonFocusListener());
 Object newValue = openDialogBox(editor);
                Object newValue = openDialogBox(editor);
 button.addFocusListener(getButtonFocusListener());
                button.addFocusListener(getButtonFocusListener());

 if (newValue != null)
                if (newValue != null)  {
{
 boolean newValidState = isCorrect(newValue);
                    boolean newValidState = isCorrect(newValue);

 if (newValidState)
                    if (newValidState)  {
{
 markDirty();
                        markDirty();
 doSetValue(newValue);
                        doSetValue(newValue);

 } else
                    } else  {
{
 setErrorMessage(MessageFormat.format(getErrorMessage(),
                        setErrorMessage(MessageFormat.format(getErrorMessage(),

 new Object[]
                                new Object[]  { newValue.toString() }));
{ newValue.toString() }));
 }
                    }
 fireApplyEditorValue();
                    fireApplyEditorValue();
 }
                }
 }
            }
 });
        });

 setValueValid(true);
        setValueValid(true);
 
        
 return editor;
        return editor;
 }
    }


 public void deactivate()
    public void deactivate()  {
{

 if (button != null && !button.isDisposed())
        if (button != null && !button.isDisposed())  {
{
 button.removeFocusListener(getButtonFocusListener());
            button.removeFocusListener(getButtonFocusListener());
 }
        }
 
        
 super.deactivate();
        super.deactivate();
 }
    }

 @Override
    @Override

 protected Object doGetValue()
    protected Object doGetValue()  {
{
 return value;
        return value;
 }
    }

 @Override
    @Override

 protected void doSetFocus()
    protected void doSetFocus()  {
{
 if (comboBox != null)
        if (comboBox != null)
 comboBox.setFocus();
            comboBox.setFocus();
 }
    }

 @Override
    @Override

 protected void doSetValue(Object value)
    protected void doSetValue(Object value)  {
{
 this.value = value;
        this.value = value;
 updateContents(value);
        updateContents(value);
 }
    }


 private void populateComboBoxItems()
    private void populateComboBoxItems()  {
{

 if (comboBox != null && items != null)
        if (comboBox != null && items != null)  {
{
 comboBox.removeAll();
            comboBox.removeAll();

 for (int i = 0; i < items.length; i++)
            for (int i = 0; i < items.length; i++)  {
{
 comboBox.add(items[i], i);
                comboBox.add(items[i], i);
 }
            }
 comboBox.setText("");
            comboBox.setText("");
 }
        }
 }
    }


 void applyEditorValueAndDeactivate()
    void applyEditorValueAndDeactivate()  {
{
 Object newValue = comboBox.getText();
        Object newValue = comboBox.getText();

 if (newValue != null && !newValue.equals(value.toString()))
        if (newValue != null && !newValue.equals(value.toString()))  {
{
 boolean newValidState = isCorrect(newValue);
            boolean newValidState = isCorrect(newValue);

 if (newValidState)
            if (newValidState)  {
{
 markDirty();
                markDirty();
 doSetValue(newValue);
                doSetValue(newValue);

 } else
            } else  {
{
 setErrorMessage(MessageFormat.format(getErrorMessage(),
                setErrorMessage(MessageFormat.format(getErrorMessage(),

 new Object[]
                        new Object[]  { newValue.toString() }));
{ newValue.toString() }));
 }
            }
 }
        }
 fireApplyEditorValue();
        fireApplyEditorValue();
 deactivate();
        deactivate();
 }
    }
 
    

 /**//*
    /**//*
 *  (non-Javadoc)
     *  (non-Javadoc)
 * @see org.eclipse.jface.viewers.CellEditor#focusLost()
     * @see org.eclipse.jface.viewers.CellEditor#focusLost()
 */
     */

 protected void focusLost()
    protected void focusLost()  {
{

 if (isActivated())
        if (isActivated())  {
{
 applyEditorValueAndDeactivate();
            applyEditorValueAndDeactivate();
 }
        }
 }
    }

 
    

 /**//*
    /**//*
 * (non-Javadoc)
     * (non-Javadoc)
 * @see org.eclipse.jface.viewers.CellEditor#keyReleaseOccured(org.eclipse.swt.events.KeyEvent)
     * @see org.eclipse.jface.viewers.CellEditor#keyReleaseOccured(org.eclipse.swt.events.KeyEvent)
 */
     */

 protected void keyReleaseOccured(KeyEvent keyEvent)
    protected void keyReleaseOccured(KeyEvent keyEvent)  {
{

 if (keyEvent.character == '\r')
        if (keyEvent.character == '\r')  { // Return key
{ // Return key
 if (comboBox != null && !comboBox.isDisposed())
            if (comboBox != null && !comboBox.isDisposed())
 fireCancelEditor();
                fireCancelEditor();

 } else if (keyEvent.character == '\t')
        } else if (keyEvent.character == '\t')  { // tab key
{ // tab key
 applyEditorValueAndDeactivate();
            applyEditorValueAndDeactivate();
 }
        }
 }
    }


 protected void editOccured(ModifyEvent e)
    protected void editOccured(ModifyEvent e)  {
{
 String value = comboBox.getText();
        String value = comboBox.getText();

 if (value == null)
        if (value == null)  {
{
 value = "";//$NON-NLS-1$
            value = "";//$NON-NLS-1$
 }
        }
 Object typedValue = value;
        Object typedValue = value;
 boolean oldValidState = isValueValid();
        boolean oldValidState = isValueValid();
 boolean newValidState = isCorrect(typedValue);
        boolean newValidState = isCorrect(typedValue);

 if (typedValue == null && newValidState)
        if (typedValue == null && newValidState)  {
{
 Assert.isTrue(false,
            Assert.isTrue(false,
 "Validator isn't limiting the cell editor's type range");//$NON-NLS-1$
                    "Validator isn't limiting the cell editor's type range");//$NON-NLS-1$
 }
        }

 if (!newValidState)
        if (!newValidState)  {
{
 // try to insert the current value into the error message.
            // try to insert the current value into the error message.
 setErrorMessage(MessageFormat.format(getErrorMessage(),
            setErrorMessage(MessageFormat.format(getErrorMessage(),

 new Object[]
                    new Object[]  { value }));
{ value }));
 }
        }
 valueChanged(oldValidState, newValidState);
        valueChanged(oldValidState, newValidState);
 }
    }


 private ModifyListener getModifyListener()
    private ModifyListener getModifyListener()  {
{

 if (modifyListener == null)
        if (modifyListener == null)  {
{

 modifyListener = new ModifyListener()
            modifyListener = new ModifyListener()  {
{

 public void modifyText(ModifyEvent e)
                public void modifyText(ModifyEvent e)  {
{
 editOccured(e);
                    editOccured(e);
 }
                }
 };
            };
 }
        }
 return modifyListener;
        return modifyListener;
 }
    }


 private FocusListener getButtonFocusListener()
    private FocusListener getButtonFocusListener()  {
{

 if (buttonFocusListener == null)
        if (buttonFocusListener == null)  {
{

 buttonFocusListener = new FocusListener()
            buttonFocusListener = new FocusListener()  {
{

 public void focusGained(FocusEvent e)
                public void focusGained(FocusEvent e)  {};
{};

 public void focusLost(FocusEvent e)
                public void focusLost(FocusEvent e)  {
{
 ComboBoxDialogCellEditor.this.focusLost();
                    ComboBoxDialogCellEditor.this.focusLost();
 }
                }
 };
            };
 };
        };
 return buttonFocusListener;
        return buttonFocusListener;
 }
    }


 private void updateContents(Object value)
    private void updateContents(Object value)  {
{
 Assert.isTrue(comboBox != null);
        Assert.isTrue(comboBox != null);
 
        

 if (value != null &&  value instanceof String)
        if (value != null &&  value instanceof String)  {
{
 comboBox.removeModifyListener(getModifyListener());
            comboBox.removeModifyListener(getModifyListener());
 comboBox.setText((String) value);
            comboBox.setText((String) value);
 comboBox.addModifyListener(getModifyListener());
            comboBox.addModifyListener(getModifyListener());
 }
        }
 }
    }
 
    
 protected abstract Object openDialogBox(Control cellEditorWindow);
    protected abstract Object openDialogBox(Control cellEditorWindow);

 }
}

posted on 2008-06-02 13:46 
自由 阅读(3947) 
评论(2)  编辑  收藏  所属分类: 
SWT